models.py 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. # -*- coding: utf-8 -*-
  2. from __future__ import unicode_literals
  3. import datetime
  4. from django.conf import settings
  5. from django.db import models
  6. from django.db.models import Q
  7. from django.core.validators import MinValueValidator
  8. class Offer(models.Model):
  9. """Description of an offer available to subscribers.
  10. Implementation notes:
  11. configuration_type store the model name of the configuration backend
  12. (ex VPNConfiguration).
  13. The choices list is dynamically generated at start in the __init__
  14. """
  15. name = models.CharField(max_length=255, blank=False, null=False,
  16. verbose_name="nom de l'offre")
  17. reference = models.CharField(max_length=255, blank=True,
  18. verbose_name="Référence de l'offre",
  19. help_text="Identifiant a utiliser par exemple comme identifiant de virement")
  20. configuration_type = models.CharField(max_length=50,
  21. blank=True,
  22. verbose_name='type de configuration',
  23. help_text="Type de configuration à utiliser avec cette offre")
  24. billing_period = models.IntegerField(blank=False, null=False, default=1,
  25. verbose_name='période de facturation',
  26. help_text='en mois',
  27. validators=[MinValueValidator(1)])
  28. period_fees = models.DecimalField(max_digits=5, decimal_places=2,
  29. blank=False, null=False,
  30. verbose_name='montant par période de '
  31. 'facturation',
  32. help_text='en €')
  33. initial_fees = models.DecimalField(max_digits=5, decimal_places=2,
  34. blank=False, null=False,
  35. verbose_name='frais de mise en service',
  36. help_text='en €')
  37. non_billable = models.BooleanField(default=False,
  38. verbose_name='n\'est pas facturable',
  39. help_text='L\'offre ne sera pas facturée par la commande charge_members')
  40. def get_configuration_type_display(self):
  41. """
  42. Renvoi le nom affichable du type de configuration
  43. """
  44. from coin.configuration.models import Configuration
  45. for item in Configuration.get_configurations_choices_list():
  46. if item and self.configuration_type in item:
  47. return item[1]
  48. return self.configuration_type
  49. get_configuration_type_display.short_description = 'type de configuration'
  50. def display_price(self):
  51. """Displays the price of an offer in a human-readable manner
  52. (for instance "30€ / month")
  53. """
  54. if int(self.period_fees) == self.period_fees:
  55. fee = int(self.period_fees)
  56. else:
  57. fee = self.period_fees
  58. if self.billing_period == 1:
  59. period = ""
  60. else:
  61. period = self.billing_period
  62. return "{period_fee}€ / {billing_period} mois".format(
  63. period_fee=fee,
  64. billing_period=period)
  65. def __unicode__(self):
  66. return '{name} - {price}'.format(name=self.name,
  67. price=self.display_price())
  68. class Meta:
  69. verbose_name = 'offre'
  70. class OfferSubscription(models.Model):
  71. """Only contains administrative details about a subscription, not
  72. technical. Nothing here should end up into the LDAP backend.
  73. Implementation notes: the Configuration model (which actually implementing the backend
  74. (technical configuration for the technology)) relate to this class
  75. with a OneToOneField
  76. """
  77. subscription_date = models.DateField(
  78. null=False,
  79. blank=False,
  80. default=datetime.date.today,
  81. verbose_name="date de souscription à l'offre")
  82. # TODO: for data retention, prevent deletion of a subscription object
  83. # while the resign date is recent enough (e.g. one year in France).
  84. resign_date = models.DateField(
  85. null=True,
  86. blank=True,
  87. verbose_name='date de résiliation')
  88. # TODO: move this to offers?
  89. commitment = models.IntegerField(blank=False, null=False,
  90. verbose_name="période d'engagement",
  91. help_text='en mois',
  92. validators=[MinValueValidator(0)],
  93. default=0)
  94. member = models.ForeignKey('members.Member', verbose_name='membre')
  95. offer = models.ForeignKey('Offer', verbose_name='offre')
  96. def get_subscription_reference(self):
  97. return settings.SUBSCRIPTION_REFERENCE.format(subscription=self)
  98. get_subscription_reference.short_description = 'Référence'
  99. def __unicode__(self):
  100. return '%s - %s - %s' % (self.member, self.offer.name,
  101. self.subscription_date)
  102. class Meta:
  103. verbose_name = 'abonnement'
  104. def count_active_subscriptions():
  105. today = datetime.date.today()
  106. query = Q(subscription_date__lte=today) & (Q(resign_date__isnull=True) | Q(resign_date__gte=today))
  107. return OfferSubscription.objects.filter(query).count()