2023-07-22 22:48:09 +00:00
|
|
|
import logging
|
|
|
|
|
2023-05-22 19:27:59 +00:00
|
|
|
from abc import (
|
|
|
|
ABC,
|
2023-07-22 22:48:09 +00:00
|
|
|
abstractmethod,
|
|
|
|
abstractproperty
|
2023-05-22 19:27:59 +00:00
|
|
|
)
|
2023-06-18 14:18:00 +00:00
|
|
|
from typing import (
|
|
|
|
List,
|
|
|
|
Any
|
|
|
|
)
|
2023-05-22 19:27:59 +00:00
|
|
|
from dataclasses import dataclass
|
|
|
|
from django.http.request import HttpRequest
|
|
|
|
from django.conf import settings
|
2023-07-22 18:39:12 +00:00
|
|
|
from django.core import signing
|
2023-05-22 19:27:59 +00:00
|
|
|
|
2023-06-18 14:18:00 +00:00
|
|
|
from store.models import (
|
|
|
|
Product,
|
2023-07-22 22:48:09 +00:00
|
|
|
ProductAuthor,
|
|
|
|
DeliveryMethod
|
2023-06-18 14:18:00 +00:00
|
|
|
)
|
2023-05-22 19:27:59 +00:00
|
|
|
|
2023-07-22 22:48:09 +00:00
|
|
|
logger = logging.getLogger("cart_logger")
|
|
|
|
|
2023-05-22 19:27:59 +00:00
|
|
|
|
|
|
|
class BaseCart(ABC):
|
|
|
|
|
2023-06-18 14:18:00 +00:00
|
|
|
def validate_and_get_product(self, item_id):
|
2023-05-22 19:27:59 +00:00
|
|
|
return Product.objects.get(id=item_id)
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def add_item(self, item_id, quantity):
|
|
|
|
...
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def remove_item(self, item_id):
|
|
|
|
...
|
|
|
|
|
|
|
|
@abstractmethod
|
|
|
|
def update_item_quantity(self, item_id, change):
|
|
|
|
...
|
|
|
|
|
2023-07-22 22:48:09 +00:00
|
|
|
@abstractproperty
|
|
|
|
def display_items(self):
|
2023-05-22 19:27:59 +00:00
|
|
|
...
|
|
|
|
|
|
|
|
|
|
|
|
class SessionCart(BaseCart):
|
|
|
|
|
2023-07-22 22:48:09 +00:00
|
|
|
|
|
|
|
def _get_author_total_price(self, author_id: int):
|
|
|
|
author_cart = self._cart[str(author_id)]
|
|
|
|
author_price = 0
|
|
|
|
product_ids = list(int(pk) for pk in author_cart.keys())
|
|
|
|
queryset = Product.objects.filter(id__in=product_ids)
|
|
|
|
for product in queryset:
|
|
|
|
author_price += product.price * author_cart[str(product.id)]
|
|
|
|
|
|
|
|
if self._delivery_info:
|
|
|
|
author_price += self._delivery_info.price
|
|
|
|
|
|
|
|
return author_price
|
|
|
|
|
|
|
|
def _prepare_display_items(self)-> List[dict[str, dict|str]]:
|
|
|
|
items: List[dict[str, dict|str]] = []
|
|
|
|
for author_id, cart_items in self._cart.items():
|
|
|
|
author = ProductAuthor.objects.get(id=int(author_id))
|
|
|
|
products = []
|
|
|
|
for item_id, quantity in cart_items.items():
|
|
|
|
product=Product.objects.get(id=int(item_id))
|
|
|
|
products.append({"product": product, "quantity": quantity})
|
|
|
|
items.append({
|
|
|
|
"author": author,
|
|
|
|
"products": products,
|
|
|
|
"group_price": self._get_author_total_price(author_id)
|
|
|
|
})
|
|
|
|
return items
|
|
|
|
|
|
|
|
def __init__(self, request: HttpRequest, delivery: DeliveryMethod=None) -> None:
|
2023-05-22 19:27:59 +00:00
|
|
|
super().__init__()
|
|
|
|
self.session = request.session
|
2023-06-18 14:18:00 +00:00
|
|
|
self._cart = self.session.get(settings.CART_SESSION_ID, None)
|
|
|
|
if not self._cart:
|
|
|
|
self._cart = {}
|
|
|
|
self.session[settings.CART_SESSION_ID] = self._cart
|
2023-07-22 22:48:09 +00:00
|
|
|
self._delivery_info = delivery
|
|
|
|
self._display_items = self._prepare_display_items()
|
|
|
|
|
2023-06-18 14:18:00 +00:00
|
|
|
def save_cart(self):
|
2023-07-22 22:48:09 +00:00
|
|
|
self._display_items = self._prepare_display_items()
|
2023-06-18 14:18:00 +00:00
|
|
|
self.session[settings.CART_SESSION_ID] = self._cart
|
|
|
|
self.session.modified = True
|
2023-05-22 19:27:59 +00:00
|
|
|
|
2023-05-22 20:11:42 +00:00
|
|
|
def add_item(self, item_id: int, quantity: int) -> None:
|
2023-06-18 14:18:00 +00:00
|
|
|
product = self.validate_and_get_product(item_id)
|
|
|
|
author = product.author
|
2023-05-28 15:06:50 +00:00
|
|
|
quantity = int(quantity)
|
|
|
|
item_id = int(item_id)
|
2023-06-18 14:18:00 +00:00
|
|
|
if not self._cart.get(str(author.id)):
|
|
|
|
self._cart[str(author.id)] = {str(item_id): quantity}
|
|
|
|
self.save_cart()
|
|
|
|
elif not self._cart[str(author.id)].get(str(item_id)):
|
|
|
|
self._cart[str(author.id)].update({str(item_id): quantity})
|
|
|
|
self.save_cart()
|
2023-05-22 19:27:59 +00:00
|
|
|
else:
|
2023-06-18 14:18:00 +00:00
|
|
|
new_quantity = self._cart[str(author.id)][str(item_id)] + quantity
|
|
|
|
self.update_item_quantity(item_id, new_quantity)
|
2023-05-22 19:27:59 +00:00
|
|
|
|
2023-05-22 20:11:42 +00:00
|
|
|
def remove_item(self, item_id: int) -> None:
|
2023-06-18 14:18:00 +00:00
|
|
|
product = self.validate_and_get_product(item_id)
|
|
|
|
author = product.author
|
2023-05-22 19:27:59 +00:00
|
|
|
try:
|
2023-06-18 14:18:00 +00:00
|
|
|
self._cart[str(author.id)].pop(str(item_id))
|
|
|
|
self.save_cart()
|
2023-05-22 19:27:59 +00:00
|
|
|
except KeyError:
|
2023-07-22 22:48:09 +00:00
|
|
|
logger.exception(f"Item {item_id} not found in cart")
|
2023-05-22 19:27:59 +00:00
|
|
|
|
2023-06-08 17:49:19 +00:00
|
|
|
def update_item_quantity(self, item_id: int, new_quantity: int) -> None:
|
2023-06-18 14:18:00 +00:00
|
|
|
product = self.validate_and_get_product(item_id)
|
|
|
|
author = product.author
|
2023-06-08 17:49:19 +00:00
|
|
|
if new_quantity < 1:
|
|
|
|
self.remove_item(item_id)
|
|
|
|
return
|
2023-06-18 14:18:00 +00:00
|
|
|
if not self._cart.get(str(author.id)):
|
2023-06-08 17:49:19 +00:00
|
|
|
self.add_item(item_id, new_quantity)
|
2023-06-18 14:18:00 +00:00
|
|
|
return
|
|
|
|
self._cart[str(author.id)][str(product.id)] = new_quantity
|
|
|
|
self.save_cart()
|
|
|
|
|
2023-07-22 22:48:09 +00:00
|
|
|
@property
|
|
|
|
def delivery_info(self):
|
|
|
|
return self._delivery_info
|
2023-05-28 15:06:50 +00:00
|
|
|
|
2023-07-22 22:48:09 +00:00
|
|
|
@property
|
|
|
|
def display_items(self) -> List[dict[str, dict|str]]:
|
|
|
|
return self._display_items
|
|
|
|
|
2023-05-28 15:06:50 +00:00
|
|
|
@property
|
|
|
|
def total_price(self):
|
|
|
|
total = 0
|
2023-06-18 14:18:00 +00:00
|
|
|
for _, cart_items in self._cart.items():
|
|
|
|
for item_id, quantity in cart_items.items():
|
|
|
|
product = Product.objects.get(id=int(item_id))
|
|
|
|
total += product.price * quantity
|
2023-07-22 22:48:09 +00:00
|
|
|
if self._delivery_info:
|
|
|
|
total += self._delivery_info.price * len(self._cart.keys())
|
2023-06-01 20:35:14 +00:00
|
|
|
return total
|
|
|
|
|
|
|
|
def is_empty(self) -> bool:
|
2023-06-18 14:18:00 +00:00
|
|
|
return not bool(self._cart.items())
|
2023-06-01 20:35:14 +00:00
|
|
|
|
|
|
|
def clear(self) -> None:
|
2023-06-18 14:18:00 +00:00
|
|
|
self._cart = {}
|
|
|
|
self.save_cart()
|
2023-07-22 18:39:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
class CustomerData:
|
|
|
|
|
|
|
|
def _encrypt_data(self, data: dict[str, Any]) -> str:
|
|
|
|
signer = signing.Signer()
|
|
|
|
return signer.sign_object(data)
|
|
|
|
|
|
|
|
def _decrypt_data(self, data: str) -> dict[str, Any]:
|
|
|
|
signer = signing.Signer()
|
|
|
|
return signer.unsign_object(data)
|
|
|
|
|
|
|
|
def __init__(self, data: dict[str, Any]=None, encrypted_data: str=None) -> None:
|
|
|
|
self._data = self._encrypt_data(data) if data else encrypted_data
|
|
|
|
|
|
|
|
@property
|
|
|
|
def data(self) -> dict[str, Any]:
|
|
|
|
return self._data
|
|
|
|
|
|
|
|
@property
|
|
|
|
def decrypted_data(self) -> dict[str, Any]:
|
|
|
|
return self._decrypt_data(self._data)
|