models.py 5.1 KB

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