fcn-report 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. #!/usr/bin/python3
  2. # Style Guide: https://www.python.org/dev/peps/pep-0008/
  3. import argparse
  4. import configparser
  5. import pprint
  6. import psycopg2
  7. # fcntoolbox package
  8. import fcntoolbox.dolibarr as dolibarr
  9. import fcntoolbox.mail as mail
  10. import fcntoolbox.sympa as sympa
  11. # network imports
  12. import smtplib
  13. from email.mime.text import MIMEText
  14. parser = argparse.ArgumentParser()
  15. parser.add_argument("-c", "--config", type=str,
  16. default="/etc/fcntoolbox/config.ini",
  17. help="specify a configuration file")
  18. args = parser.parse_args()
  19. config = configparser.RawConfigParser()
  20. config.sections()
  21. config.read(args.config)
  22. confac = config['ac']
  23. confdoli = config['dolibarr']
  24. confsympa = config['sympa']
  25. confsmtp = config['smtp']
  26. conn = psycopg2.connect(host=confdoli['host'], database=confdoli['database'],
  27. user=confdoli['user'], password=confdoli['password'])
  28. doli = dolibarr.Instance(conn)
  29. s = smtplib.SMTP(confsmtp.get('host', 'localhost'))
  30. pp = pprint.PrettyPrinter()
  31. if confac.get('audit_dolibarr_users', False):
  32. usrfields = ["rowid", "firstname", "lastname", "email"]
  33. cur = doli.get_users(fields=usrfields, active_user=True, active_member=False)
  34. badusers = cur.fetchall()
  35. # Exception for user id = 1 (default admin user)
  36. badusers = list(filter(lambda usr: usr[0] != 1, badusers))
  37. if len(badusers) != 0:
  38. body = "Salut, \n\nDes utilisateurs ne sont pas associés à un membre actif : \n\n" + pp.pformat(badusers)
  39. msg = MIMEText(body)
  40. msg['Subject'] = "Rapport utilisateurs Dolibarr"
  41. msg['From'] = confsmtp.get('from', 'root')
  42. msg['To'] = confac.get('audit_recipients', 'root')
  43. s.send_message(msg)
  44. fields = ["rowid", "firstname", "lastname", "email"]
  45. fieldsextra = ['email_second']
  46. # Filter out missing extra fields
  47. #cur.execute("SELECT * FROM llx_adherent_extrafields LIMIT 1")
  48. #columns_extra = map(lambda col : col.name, cur.description)
  49. #rows_extra = filter(lambda r : r in columns_extra, fieldsextra)
  50. cur = doli.get_adherent(fields, fieldsextra)
  51. aliases = mail.Aliases('/etc/aliases', 'franciliens.net')
  52. raliases = dict((val, key) for key, val in aliases.items())
  53. adherents = set() # set(email)
  54. adherents_alt = dict() # dict(email => set(email))
  55. radherents_alt = dict() # dict(email => email)
  56. for record in cur:
  57. columns = map(lambda col : col.name, cur.description)
  58. adh = dict(zip(columns, record))
  59. adh_email = adh['email'].lower()
  60. adherents.add(adh_email)
  61. adh_alt = set()
  62. if 'email_second' in adh and adh['email_second'] != None:
  63. adh_alt.add(adh['email_second'])
  64. radherents_alt[adh['email_second']] = adh_email
  65. if adh_email in aliases:
  66. adh_alt.add(aliases[adh_email])
  67. radherents_alt[aliases[adh_email]] = adh_email
  68. if adh_email in raliases:
  69. adh_alt.add(raliases[adh_email])
  70. radherents_alt[raliases[adh_email]] = adh_email
  71. adherents_alt[adh_email] = adh_alt
  72. sympaconn = psycopg2.connect(host=confsympa['host'], database=confsympa['database'],
  73. user=confsympa['user'], password=confsympa['password'])
  74. sympaIns = sympa.Instance(sympaconn)
  75. list = 'adherents'
  76. francisubs = set(sympaIns.get_subscribers(list, confsympa['robot']))
  77. tosubscribe = set()
  78. for adh in adherents:
  79. adh_alt = adherents_alt[adh]
  80. if adh in francisubs or len(adh_alt.intersection(francisubs)) > 0:
  81. pass
  82. else:
  83. tosubscribe.add(adh)
  84. tounsubscribe = set()
  85. for sub in francisubs:
  86. if sub in adherents \
  87. or (sub in radherents_alt and radherents_alt[sub] in adherents):
  88. pass
  89. else:
  90. tounsubscribe.add(sub)
  91. editors = set(sympaIns.get_editors(list, confsympa['robot']))
  92. remeditors = set()
  93. for editor in editors:
  94. if editor in adherents \
  95. or (editor in radherents_alt and radherents_alt[editor] in adherents):
  96. pass
  97. else:
  98. remeditors.add(editor)
  99. reportFormat = open('/etc/fcntoolbox/fcn-report.format', 'r').read()
  100. def sendReport(listname, tosubscribe, tounsubscribe, removeeditors):
  101. body = reportFormat.format(
  102. list = listname,
  103. tosubscribe = pp.pformat(tosubscribe),
  104. removesubscribers = pp.pformat(tounsubscribe),
  105. removeeditors = pp.pformat(removeeditors))
  106. msg = MIMEText(body)
  107. msg['Subject'] = "Rapport de la liste [%s]" % listname
  108. msg['From'] = confsympa['mail_from']
  109. if 'mail_cc' in confsympa:
  110. msg['Cc'] = confsympa['mail_cc']
  111. msg['To'] = "%s-request@%s" % (listname, confsympa['robot'])
  112. s.send_message(msg)
  113. if len(adherents) > 0 and (len(tosubscribe) > 0 or len(tounsubscribe) > 0) :
  114. sendReport(list, tosubscribe, tounsubscribe, remeditors)
  115. molists = confac['members_only_lists'].split(',')
  116. for list in molists:
  117. subs = set(sympaIns.get_subscribers(list, confsympa['robot']))
  118. tounsubscribe = set()
  119. for sub in subs:
  120. if sub in adherents \
  121. or (sub in radherents_alt and radherents_alt[sub] in adherents):
  122. pass
  123. else:
  124. tounsubscribe.add(sub)
  125. editors = set(sympaIns.get_editors(list, confsympa['robot']))
  126. remeditors = set()
  127. for editor in editors:
  128. if editor in adherents \
  129. or (editor in radherents_alt and radherents_alt[editor] in adherents):
  130. pass
  131. else:
  132. remeditors.add(editor)
  133. if len(adherents) > 0 and len(tounsubscribe) > 0 :
  134. sendReport(list, {}, tounsubscribe, remeditors)
  135. s.quit()