1
0

models.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. # -*- coding: utf-8 -*-
  2. import ldapdb.models
  3. import pprint
  4. import os
  5. import base64
  6. import hashlib
  7. import unicodedata
  8. import string
  9. import datetime
  10. from django.db import models
  11. from django.db.models import Q
  12. from django.db.models.signals import post_save, pre_save, post_delete
  13. from django.dispatch import receiver
  14. from django.core import exceptions
  15. from ldapdb.models.fields import CharField, IntegerField, ListField
  16. from south.modelsinspector import add_ignored_fields
  17. from coin.offers.models import OfferSubscription
  18. from coin.models import CoinLdapSyncModel
  19. from django.contrib.auth.signals import user_logged_in
  20. from django.conf import settings
  21. class Member(CoinLdapSyncModel):
  22. MEMBER_TYPE_CHOICES = (
  23. ('personne_physique', 'Personne physique'),
  24. ('personne_morale', 'Personne morale'),
  25. )
  26. MEMBER_STATUS_CHOICES = (
  27. ('adherent', 'Adhérent'),
  28. ('non_adherent', 'Non adhérent'),
  29. ('demande_adhesion', "Demande d'adhésion"),
  30. )
  31. user = models.OneToOneField(settings.AUTH_USER_MODEL, null=True, default=None,
  32. verbose_name='Utilisateur Django',
  33. on_delete=models.SET_NULL)
  34. status = models.CharField(max_length=50, choices=MEMBER_STATUS_CHOICES,
  35. default='non_adherent')
  36. type = models.CharField(max_length=20, choices=MEMBER_TYPE_CHOICES,
  37. default='personne_physique')
  38. first_name = models.CharField(max_length=200, verbose_name=u'Prénom')
  39. last_name = models.CharField(max_length=200, verbose_name=u'Nom')
  40. ldap_cn = models.CharField(max_length=200, blank=True,
  41. help_text='Clé avec le LDAP. Laisser vide pour '
  42. 'la générer automatiquement')
  43. organization_name = models.CharField(max_length=200, blank=True,
  44. verbose_name='Nom de l\'organisme',
  45. help_text='Pour une personne morale')
  46. email = models.EmailField(max_length=254, verbose_name=u'Courriel')
  47. home_phone_number = models.CharField(max_length=25, blank=True,
  48. verbose_name=u'Téléphone fixe')
  49. mobile_phone_number = models.CharField(max_length=25, blank=True,
  50. verbose_name=u'Téléphone mobile')
  51. address = models.TextField(verbose_name=u'Adresse')
  52. postal_code = models.CharField(max_length=15,
  53. verbose_name=u'Code postal')
  54. city = models.CharField(max_length=200,
  55. verbose_name=u'Commune')
  56. country = models.CharField(max_length=200,
  57. default='France',
  58. verbose_name=u'Pays')
  59. entry_date = models.DateField(null=False,
  60. blank=False,
  61. default=datetime.date.today,
  62. verbose_name='Date de première adhésion')
  63. resign_date = models.DateField(null=True, blank=True,
  64. verbose_name='Date de départ de '
  65. 'l\'association')
  66. def __unicode__(self):
  67. name = self.first_name + ' ' + self.last_name
  68. if (self.organization_name):
  69. name += ' (%s)' % self.organization_name
  70. return name
  71. # Renvoi la date de fin de la dernière cotisation du membre
  72. def end_date_of_membership(self):
  73. try:
  74. return self.membership_fees.order_by('-end_date')[0].end_date
  75. except:
  76. return None
  77. def change_password(self, new_password):
  78. ldap_user = LdapUser.objects.get(pk=self.ldap_cn)
  79. ldap_user.password = new_password
  80. ldap_user.save()
  81. def get_active_subscriptions(self, date=datetime.date.today()):
  82. return OfferSubscription.objects.filter(
  83. Q(member__exact=self.pk),
  84. Q(subscription_date__lte=date),
  85. Q(resign_date__isnull=True) | Q(resign_date__gte=date))
  86. def get_automatic_ldap_cn(self):
  87. """
  88. Calcul le login / ldap_cn automatiquement en fonction du nom et du prénom
  89. """
  90. # Première lettre de chaque partie du prénom
  91. first_name_letters = ''.join(
  92. [c[0] for c in self.first_name.split('-')]
  93. )
  94. # Concaténer avec nom de famille
  95. ldap_cn = ('%s%s' % (first_name_letters, self.last_name))
  96. # Remplacer ou enlever les caractères non ascii
  97. ldap_cn = unicodedata.normalize('NFD', ldap_cn)\
  98. .encode('ascii', 'ignore')
  99. # Enlever ponctuation et espace
  100. ldap_cn = ldap_cn.translate(None, string.punctuation + ' ')
  101. # En minuscule
  102. ldap_cn = ldap_cn.lower()
  103. return ldap_cn
  104. def sync_to_ldap(self, creation):
  105. """
  106. Update LDAP data when a member is saved
  107. """
  108. assert self.ldap_cn, ('Can\'t sync with LDAP because missing ldap_cn '
  109. 'value for the Member : %s' % self)
  110. if not creation:
  111. ldap_user = LdapUser.objects.get(pk=self.ldap_cn)
  112. if creation:
  113. max_uidNumber = LdapUser.objects.order_by('-uidNumber')[0].uidNumber
  114. ldap_user = LdapUser()
  115. ldap_user.pk = self.ldap_cn
  116. ldap_user.uid = self.ldap_cn
  117. ldap_user.nick_name = self.ldap_cn
  118. ldap_user.uidNumber = max_uidNumber + 1
  119. ldap_user.last_name = self.last_name
  120. ldap_user.first_name = self.first_name
  121. ldap_user.save()
  122. if creation:
  123. ldap_group = LdapGroup.objects.get(pk='coin')
  124. ldap_group.members.append(ldap_user.pk)
  125. ldap_group.save()
  126. class Meta:
  127. verbose_name = 'membre'
  128. class CryptoKey(models.Model):
  129. KEY_TYPE_CHOICES = (('RSA', 'RSA'), ('GPG', 'GPG'))
  130. type = models.CharField(max_length=3, choices=KEY_TYPE_CHOICES)
  131. key = models.TextField(verbose_name=u'Clé')
  132. member = models.ForeignKey('Member', verbose_name=u'Membre')
  133. def __unicode__(self):
  134. return u'Clé %s de %s' % (self.type, self.member)
  135. class Meta:
  136. verbose_name = 'clé'
  137. class MembershipFee(models.Model):
  138. member = models.ForeignKey('Member', related_name='membership_fees',
  139. verbose_name=u'Membre')
  140. amount = models.IntegerField(null=False, default='20', help_text='en €',
  141. verbose_name=u'Montant')
  142. start_date = models.DateField(
  143. null=False,
  144. blank=False,
  145. default=datetime.date.today,
  146. verbose_name='Date de début de cotisation')
  147. end_date = models.DateField(
  148. null=False,
  149. blank=False,
  150. default=datetime.date.today() + datetime.timedelta(365),
  151. verbose_name='Date de fin de cotisation')
  152. def __unicode__(self):
  153. return (u'%s - %s - %i€' % (self.member, self.start_date,
  154. self.amount))
  155. class Meta:
  156. verbose_name = 'cotisation'
  157. class LdapUser(ldapdb.models.Model):
  158. # TODO: déplacer ligne suivante dans settings.py
  159. base_dn = "ou=users,ou=unix,o=ILLYSE,l=Villeurbanne,st=RHA,c=FR"
  160. object_classes = ['inetOrgPerson', 'organizationalPerson', 'person',
  161. 'top', 'posixAccount']
  162. uid = CharField(db_column='uid', unique=True, max_length=255)
  163. nick_name = CharField(db_column='cn', unique=True, primary_key=True,
  164. max_length=255)
  165. first_name = CharField(db_column='givenName', max_length=255)
  166. last_name = CharField(db_column='sn', max_length=255)
  167. display_name = CharField(db_column='displayName', max_length=255,
  168. blank=True)
  169. password = CharField(db_column='userPassword', max_length=255)
  170. uidNumber = IntegerField(db_column='uidNumber', unique=True)
  171. gidNumber = IntegerField(db_column='gidNumber', default=2000)
  172. homeDirectory = CharField(db_column='homeDirectory', max_length=255,
  173. default='/tmp')
  174. def __unicode__(self):
  175. return self.display_name
  176. class Meta:
  177. managed = False # Indique à South de ne pas gérer le model LdapUser
  178. class LdapGroup(ldapdb.models.Model):
  179. """
  180. Class for representing an LDAP group entry.
  181. """
  182. # LDAP meta-data
  183. base_dn = "ou=groups,ou=unix,o=ILLYSE,l=Villeurbanne,st=RHA,c=FR"
  184. object_classes = ['posixGroup']
  185. # posixGroup attributes
  186. gid = IntegerField(db_column='gidNumber', unique=True)
  187. name = CharField(db_column='cn', max_length=200, primary_key=True)
  188. members = ListField(db_column='memberUid')
  189. def __unicode__(self):
  190. return self.name
  191. class Meta:
  192. managed = False # Indique à South de ne pas gérer le model LdapGroup
  193. # Indique à South de ne pas gérer les models LdapUser et LdapGroup
  194. add_ignored_fields(["^ldapdb\.models\.fields"])
  195. @receiver(pre_save, sender=Member)
  196. def define_ldap_cn(sender, instance, **kwargs):
  197. """
  198. Lors de la sauvegarde d'un membre. Si le champ ldap_cn n'est pas définit,
  199. le calcul automatiquement en fonction du nom et du prénom
  200. """
  201. if not instance.ldap_cn and not instance.pk:
  202. instance.ldap_cn = instance.get_automatic_ldap_cn()
  203. @receiver(pre_save, sender=LdapUser)
  204. def change_password(sender, instance, **kwargs):
  205. """
  206. Lors de la sauvegarde d'un utilisateur Ldap, cette fonction est exécutée
  207. avant la sauvegarde pour chiffrer le mot de passe s'il est définit
  208. et s'il n'est pas déjà chiffré
  209. """
  210. # Si le mot de passe est définit et n'est pas déjà chiffré,
  211. # alors ça le chiffre
  212. if instance.password and not instance.password.startswith('{SSHA}'):
  213. salt = os.urandom(8).encode('hex')
  214. digest = hashlib.sha1(instance.password + salt).digest()
  215. instance.password = '{SSHA}' + base64.b64encode(digest + salt)
  216. @receiver(pre_save, sender=LdapUser)
  217. def define_display_name(sender, instance, **kwargs):
  218. """
  219. Lors de la sauvegarde d'un utilisateur Ldap, le champ display_name est la
  220. concaténation de first_name et last_name
  221. """
  222. if not instance.display_name:
  223. instance.display_name = '%s %s' % (instance.first_name,
  224. instance.last_name)
  225. @receiver(post_delete, sender=Member)
  226. def remove_ldap_user_from_coin_group_when_deleting_member(sender,
  227. instance,
  228. **kwargs):
  229. """
  230. Lorsqu'un membre est supprimé du SI, son utilisateur LDAP correspondant est
  231. sorti du groupe "coin"
  232. """
  233. ldap_group = LdapGroup.objects.get(pk='coin')
  234. if instance.ldap_cn in ldap_group.members:
  235. ldap_group.members.remove(instance.ldap_cn)
  236. ldap_group.save()
  237. @receiver(user_logged_in)
  238. def define_member_user(sender, request, user, **kwargs):
  239. """
  240. Lorsqu'un utilisateur se connect avec succes, fait le lien entre le membre
  241. et l'utilisateur en définissant le champ user du model membre ayant le
  242. ldap_cn utilisé pour la connexion
  243. """
  244. try:
  245. member = Member.objects.get(ldap_cn=user.username)
  246. if not member.user:
  247. member.user = user
  248. member.save()
  249. elif member.user.username != user.username:
  250. raise Exception('Un membre avec cet ldap_cn existe en base de donnée '
  251. 'mais l\'utilisateur auquel il est rattaché ne '
  252. 'correspond pas.')
  253. except Member.DoesNotExist:
  254. if not user.is_superuser:
  255. raise
  256. #==============================================================================
  257. # @receiver(pre_save, sender = LdapUser)
  258. # def ssha_password(sender, **kwargs):
  259. # if not kwargs['instance'].password.startswith('{SSHA}'):
  260. # salt = os.urandom(8).encode('hex')
  261. # kwargs['instance'].password = '{SSHA}' + base64.b64encode(
  262. # hashlib.sha1(obj.password + salt).digest() + salt)
  263. #==============================================================================