checkout/src/purchase/models.py

250 lines
6.6 KiB
Python

from __future__ import annotations
import uuid
from django.core.validators import MaxValueValidator
from django.db import models
from django.db.models import Avg, Count, F, Sum
from django.db.models.functions import Coalesce
from django.urls import reverse
from django.utils.translation import gettext
from django.utils.translation import gettext_lazy as _
from solo.models import SingletonModel
class Model(models.Model):
created_at = models.DateTimeField(auto_now_add=True, verbose_name=_("created at"))
updated_at = models.DateTimeField(auto_now=True, verbose_name=_("updated at"))
class Meta:
abstract = True
class PaymentMethodQuerySet(models.QuerySet):
def with_turnover(self):
return self.annotate(
turnover=Coalesce(
Sum(
F("baskets__items__quantity")
* F("baskets__items__unit_price_cents"),
),
0,
),
)
def with_sold(self):
return self.annotate(sold=Count("baskets", distinct=True))
class PaymentMethodManager(models.Manager):
def get_by_natural_key(self, name):
return self.get(name=name)
class PaymentMethod(Model):
name = models.CharField(max_length=50, unique=True, verbose_name=_("name"))
objects = PaymentMethodManager.from_queryset(PaymentMethodQuerySet)()
class Meta:
verbose_name = _("payment method")
verbose_name_plural = _("payment methods")
ordering = ("name",)
def __str__(self):
return self.name
def natural_key(self):
return (self.name,)
def default_product_display_order():
last = Product.objects.last()
if last is None:
return 1
return last.display_order + 1
class ProductCategory(Model):
name = models.CharField(max_length=250, unique=True, verbose_name=_("name"))
color_hue = models.PositiveIntegerField(
validators=[MaxValueValidator(360)],
verbose_name=_("color hue"),
help_text=_("Color hue in degrees (0-360)"),
)
class Meta:
verbose_name = _("product category")
verbose_name_plural = _("product categories")
def __str__(self):
return self.name
class ProductQuerySet(models.QuerySet):
def with_turnover(self):
return self.annotate(
turnover=Coalesce(
Sum(F("basket_items__quantity") * F("basket_items__unit_price_cents")),
0,
),
)
def with_sold(self):
return self.annotate(sold=Coalesce(Sum("basket_items__quantity"), 0))
def with_fixed_price(self):
return self.exclude(unit_price_cents=0)
def with_no_fixed_price(self):
return self.filter(unit_price_cents=0)
def with_category(self):
return self.select_related("category")
class ProductManager(models.Manager):
def get_by_natural_key(self, name):
return self.get(name=name)
class Product(Model):
name = models.CharField(max_length=250, unique=True, verbose_name=_("name"))
category = models.ForeignKey(
ProductCategory,
on_delete=models.PROTECT,
verbose_name=_("category"),
null=False,
blank=False,
)
unit_price_cents = models.PositiveIntegerField(
verbose_name=_("unit price (cents)"),
help_text=_(
"Unit price in cents. Use zero to denote that the product has no fixed price.",
),
)
initials = models.CharField(
max_length=10,
verbose_name=_("initials"),
blank=False,
null=False,
)
display_order = models.PositiveIntegerField(
default=default_product_display_order,
verbose_name=_("display order"),
)
objects = ProductManager.from_queryset(ProductQuerySet)()
class Meta:
ordering = ["display_order", "name"]
verbose_name = _("product")
verbose_name_plural = _("products")
def __str__(self):
return self.name
def natural_key(self):
return (self.name,)
@property
def color_hue(self):
return self.category.color_hue
@property
def has_fixed_price(self) -> bool:
return self.unit_price_cents > 0
class BasketQuerySet(models.QuerySet):
def with_articles_count(self) -> BasketQuerySet:
return self.annotate(articles_count=Sum(F("items__quantity")))
def priced(self) -> BasketQuerySet:
return self.annotate(
price=Coalesce(Sum(F("items__quantity") * F("items__unit_price_cents")), 0),
)
def average_basket(self) -> float:
return self.priced().aggregate(avg=Avg("price"))["avg"]
def by_date(self, date) -> BasketQuerySet:
return self.filter(created_at__date=date)
def turnover(self) -> int:
return self.priced().aggregate(total=Sum("price"))["total"]
class Basket(Model):
payment_method = models.ForeignKey(
to=PaymentMethod,
on_delete=models.PROTECT,
related_name="baskets",
null=False,
blank=False,
verbose_name=_("payment method"),
)
objects = BasketQuerySet.as_manager()
class Meta:
verbose_name = _("basket")
verbose_name_plural = _("baskets")
def __str__(self):
return gettext("Basket #%(id)s") % {"id": self.id}
def get_absolute_url(self):
return reverse("purchase:update", args=(self.pk,))
class BasketItemQuerySet(models.QuerySet):
def priced(self):
return self.annotate(price=Coalesce(F("quantity") * F("unit_price_cents"), 0))
class BasketItem(Model): # noqa: DJ008
product = models.ForeignKey(
to=Product,
on_delete=models.PROTECT,
related_name="basket_items",
verbose_name=_("product"),
)
basket = models.ForeignKey(
to=Basket,
on_delete=models.CASCADE,
related_name="items",
verbose_name=_("basket"),
)
quantity = models.PositiveIntegerField(verbose_name=_("quantity"))
unit_price_cents = models.PositiveIntegerField(
verbose_name=_("unit price (cents)"),
help_text=_("product's unit price in cents at the time of purchase"),
)
objects = BasketItemQuerySet.as_manager()
class Meta:
verbose_name = _("basket item")
verbose_name_plural = _("basket items")
class Cache(SingletonModel):
etag = models.UUIDField(default=uuid.uuid4)
last_modified = models.DateTimeField(auto_now=True)
def __str__(self) -> str:
return str(self.etag)
def refresh(self):
self.etag = uuid.uuid4()
self.save()
def reports_etag(_request):
return str(Cache.get_solo().etag)
def reports_last_modified(_request):
return Cache.get_solo().last_modified