django-orchestra/orchestra/contrib/plans/models.py

103 lines
3.6 KiB
Python
Raw Normal View History

2014-11-18 13:59:21 +00:00
from django.core.validators import ValidationError
from django.db import models
2015-03-27 19:50:54 +00:00
from django.db.models import Q
2015-04-07 15:14:49 +00:00
from django.utils.functional import cached_property
2014-11-18 13:59:21 +00:00
from django.utils.translation import ugettext_lazy as _
from orchestra.core.validators import validate_name
from orchestra.models import queryset
from . import rating
class Plan(models.Model):
name = models.CharField(_("name"), max_length=32, unique=True, validators=[validate_name])
verbose_name = models.CharField(_("verbose_name"), max_length=128, blank=True)
2015-04-08 14:41:09 +00:00
is_active = models.BooleanField(_("active"), default=True,
help_text=_("Designates whether this account should be treated as active. "
"Unselect this instead of deleting accounts."))
2014-11-18 13:59:21 +00:00
is_default = models.BooleanField(_("default"), default=False,
help_text=_("Designates whether this plan is used by default or not."))
is_combinable = models.BooleanField(_("combinable"), default=True,
help_text=_("Designates whether this plan can be combined with other plans or not."))
allow_multiple = models.BooleanField(_("allow multiple"), default=False,
help_text=_("Designates whether this plan allow for multiple contractions."))
2015-04-02 16:14:55 +00:00
def __str__(self):
2015-03-27 19:50:54 +00:00
return self.get_verbose_name()
2014-11-18 13:59:21 +00:00
def clean(self):
self.verbose_name = self.verbose_name.strip()
def get_verbose_name(self):
return self.verbose_name or self.name
class ContractedPlan(models.Model):
plan = models.ForeignKey(Plan, verbose_name=_("plan"), related_name='contracts')
account = models.ForeignKey('accounts.Account', verbose_name=_("account"),
2015-04-05 10:46:24 +00:00
related_name='plans')
2014-11-18 13:59:21 +00:00
class Meta:
verbose_name_plural = _("plans")
2015-04-02 16:14:55 +00:00
def __str__(self):
2014-11-18 13:59:21 +00:00
return str(self.plan)
2015-04-07 15:14:49 +00:00
@cached_property
def active(self):
return self.plan.is_active and self.account.is_active
2014-11-18 13:59:21 +00:00
def clean(self):
2015-03-27 19:50:54 +00:00
if not self.pk and not self.plan.allow_multiple:
2014-11-18 13:59:21 +00:00
if ContractedPlan.objects.filter(plan=self.plan, account=self.account).exists():
raise ValidationError("A contracted plan for this account already exists.")
class RateQuerySet(models.QuerySet):
group_by = queryset.group_by
def by_account(self, account):
# Default allways selected
return self.filter(
Q(plan__is_default=True) |
Q(plan__contracts__account=account)
).order_by('plan', 'quantity').select_related('plan')
class Rate(models.Model):
STEP_PRICE = 'STEP_PRICE'
MATCH_PRICE = 'MATCH_PRICE'
BEST_PRICE = 'BEST_PRICE'
2014-11-18 13:59:21 +00:00
RATE_METHODS = {
STEP_PRICE: rating.step_price,
MATCH_PRICE: rating.match_price,
BEST_PRICE: rating.best_price,
2014-11-18 13:59:21 +00:00
}
service = models.ForeignKey('services.Service', verbose_name=_("service"),
2015-04-05 10:46:24 +00:00
related_name='rates')
2014-11-18 13:59:21 +00:00
plan = models.ForeignKey(Plan, verbose_name=_("plan"), related_name='rates')
2015-04-24 14:03:42 +00:00
quantity = models.PositiveIntegerField(_("quantity"), null=True, blank=True,
help_text=_("See rate algorihm help text."))
2014-11-18 13:59:21 +00:00
price = models.DecimalField(_("price"), max_digits=12, decimal_places=2)
objects = RateQuerySet.as_manager()
class Meta:
unique_together = ('service', 'plan', 'quantity')
2015-04-02 16:14:55 +00:00
def __str__(self):
2014-11-18 13:59:21 +00:00
return "{}-{}".format(str(self.price), self.quantity)
@classmethod
def get_methods(cls):
return cls.RATE_METHODS
@classmethod
def get_choices(cls):
choices = []
2015-04-02 16:14:55 +00:00
for name, method in cls.RATE_METHODS.items():
choices.append((name, method.verbose_name))
return choices