from decimal import Decimal from typing import Any, Dict, Iterable, List, Tuple from django.db.models import Count, Sum from django.utils.translation import ugettext_lazy as _ from pretix.base.models import Event, Item, ItemCategory, Order, OrderPosition from pretix.base.models.event import SubEvent from pretix.base.models.orders import OrderFee class DummyObject: pass class Dontsum: def __init__(self, value: Any): self.value = value def __str__(self) -> str: return str(self.value) def tuplesum(tuples: Iterable[Tuple]) -> Tuple: """ Takes a list of tuples of size n. In our case, those are e.g. tuples of size 2 containing a number of sales and a sum of their toal amount. Returned is again a tuple of size n. The first component of the returned tuple is the sum of the first components of all input tuples. Sample: >>> tuplesum([(1, 2), (3, 4), (5, 6)]) (9, 12) """ def mysum(it): # This method is identical to sum(list), except that it ignores entries of the type # Dontsum. We need this because we list the payment method fees seperately but we don't # want a order to contribute twice to the total count of orders (once for a product # and once for the payment method fee). sit = [i for i in it if not isinstance(i, Dontsum)] return sum(sit) # zip(*list(tuples)) basically transposes our input, e.g. [(1,2), (3,4), (5,6)] # becomes [(1, 3, 5), (2, 4, 6)]. We then call map on that, such that mysum((1, 3, 5)) # and mysum((2, 4, 6)) will be called. The results will then be combined in a tuple again. return tuple(map(mysum, zip(*list(tuples)))) def dictsum(*dicts) -> dict: """ Takes multiple dictionaries as arguments and builds a new dict. The input dict is expected to be a mapping of keys to tuples. The output dict will contain all keys that are present in any of the input dicts and will contain the tuplesum of all values associated with this key (see tuplesum function). Sample: >>> dictsum({'a': (1, 2), 'b': (3, 4)}, {'a': (5, 6), 'c': (7, 8)}) {'a': (6, 8), 'b': (3, 4), 'c': (7, 8)} """ res = {} keys = set() for d in dicts: keys |= set(d.keys()) for k in keys: res[k] = tuplesum(d[k] for d in dicts if k in d) return res def order_overview(event: Event, subevent: SubEvent=None) -> Tuple[List[Tuple[ItemCategory, List[Item]]], Dict[str, Tuple[Decimal, Decimal]]]: items = event.items.all().select_related( 'category', # for re-grouping ).prefetch_related( 'variations' ).order_by('category__position', 'category_id', 'position', 'name') qs = OrderPosition.objects if subevent: qs = qs.filter(subevent=subevent) counters = qs.filter( order__event=event ).values( 'item', 'variation', 'order__status' ).annotate(cnt=Count('id'), price=Sum('price'), tax_value=Sum('tax_value')).order_by() num_canceled = { (p['item'], p['variation']): (p['cnt'], p['price'], p['price'] - p['tax_value']) for p in counters if p['order__status'] == Order.STATUS_CANCELED } num_refunded = { (p['item'], p['variation']): (p['cnt'], p['price'], p['price'] - p['tax_value']) for p in counters if p['order__status'] == Order.STATUS_REFUNDED } num_paid = { (p['item'], p['variation']): (p['cnt'], p['price'], p['price'] - p['tax_value']) for p in counters if p['order__status'] == Order.STATUS_PAID } num_pending = { (p['item'], p['variation']): (p['cnt'], p['price'], p['price'] - p['tax_value']) for p in counters if p['order__status'] == Order.STATUS_PENDING } num_expired = { (p['item'], p['variation']): (p['cnt'], p['price'], p['price'] - p['tax_value']) for p in counters if p['order__status'] == Order.STATUS_EXPIRED } num_total = dictsum(num_pending, num_paid) for item in items: item.all_variations = list(item.variations.all()) item.has_variations = (len(item.all_variations) > 0) if item.has_variations: for var in item.all_variations: variid = var.id var.num_total = num_total.get((item.id, variid), (0, 0, 0)) var.num_pending = num_pending.get((item.id, variid), (0, 0, 0)) var.num_expired = num_expired.get((item.id, variid), (0, 0, 0)) var.num_canceled = num_canceled.get((item.id, variid), (0, 0, 0)) var.num_refunded = num_refunded.get((item.id, variid), (0, 0, 0)) var.num_paid = num_paid.get((item.id, variid), (0, 0, 0)) item.num_total = tuplesum(var.num_total for var in item.all_variations) item.num_pending = tuplesum(var.num_pending for var in item.all_variations) item.num_expired = tuplesum(var.num_expired for var in item.all_variations) item.num_canceled = tuplesum(var.num_canceled for var in item.all_variations) item.num_refunded = tuplesum(var.num_refunded for var in item.all_variations) item.num_paid = tuplesum(var.num_paid for var in item.all_variations) else: item.num_total = num_total.get((item.id, None), (0, 0, 0)) item.num_pending = num_pending.get((item.id, None), (0, 0, 0)) item.num_expired = num_expired.get((item.id, None), (0, 0, 0)) item.num_canceled = num_canceled.get((item.id, None), (0, 0, 0)) item.num_refunded = num_refunded.get((item.id, None), (0, 0, 0)) item.num_paid = num_paid.get((item.id, None), (0, 0, 0)) nonecat = ItemCategory(name=_('Uncategorized')) # Regroup those by category items_by_category = sorted( [ # a group is a tuple of a category and a list of items (cat if cat is not None else nonecat, [i for i in items if i.category == cat]) for cat in set([i.category for i in items]) # insert categories into a set for uniqueness # a set is unsorted, so sort again by category ], key=lambda group: (group[0].position, group[0].id) if ( group[0] is not None and group[0].id is not None) else (0, 0) ) for c in items_by_category: c[0].num_total = tuplesum(item.num_total for item in c[1]) c[0].num_pending = tuplesum(item.num_pending for item in c[1]) c[0].num_expired = tuplesum(item.num_expired for item in c[1]) c[0].num_canceled = tuplesum(item.num_canceled for item in c[1]) c[0].num_refunded = tuplesum(item.num_refunded for item in c[1]) c[0].num_paid = tuplesum(item.num_paid for item in c[1]) # Payment fees payment_cat_obj = DummyObject() payment_cat_obj.name = _('Fees') payment_items = [] if not subevent: counters = OrderFee.objects.filter( order__event=event ).values( 'fee_type', 'internal_type', 'order__status' ).annotate(cnt=Count('id'), value=Sum('value'), tax_value=Sum('tax_value')).order_by() num_canceled = { (o['fee_type'], o['internal_type']): (o['cnt'], o['value'], o['value'] - o['tax_value']) for o in counters if o['order__status'] == Order.STATUS_CANCELED } num_refunded = { (o['fee_type'], o['internal_type']): (o['cnt'], o['value'], o['value'] - o['tax_value']) for o in counters if o['order__status'] == Order.STATUS_REFUNDED } num_pending = { (o['fee_type'], o['internal_type']): (o['cnt'], o['value'], o['value'] - o['tax_value']) for o in counters if o['order__status'] == Order.STATUS_PENDING } num_expired = { (o['fee_type'], o['internal_type']): (o['cnt'], o['value'], o['value'] - o['tax_value']) for o in counters if o['order__status'] == Order.STATUS_EXPIRED } num_paid = { (o['fee_type'], o['internal_type']): (o['cnt'], o['value'], o['value'] - o['tax_value']) for o in counters if o['order__status'] == Order.STATUS_PAID } num_total = dictsum(num_pending, num_paid) provider_names = { k: v.verbose_name for k, v in event.get_payment_providers().items() } names = dict(OrderFee.FEE_TYPES) for pprov, total in num_total.items(): ppobj = DummyObject() if pprov[0] == OrderFee.FEE_TYPE_PAYMENT: ppobj.name = '{} - {}'.format(names[OrderFee.FEE_TYPE_PAYMENT], provider_names.get(pprov[1], pprov[1])) else: ppobj.name = '{} - {}'.format(names[OrderFee.FEE_TYPE_PAYMENT], pprov[1]) ppobj.provider = pprov[1] ppobj.has_variations = False ppobj.num_total = total ppobj.num_canceled = num_canceled.get(pprov, (0, 0, 0)) ppobj.num_refunded = num_refunded.get(pprov, (0, 0, 0)) ppobj.num_expired = num_expired.get(pprov, (0, 0, 0)) ppobj.num_pending = num_pending.get(pprov, (0, 0, 0)) ppobj.num_paid = num_paid.get(pprov, (0, 0, 0)) payment_items.append(ppobj) payment_cat_obj.num_total = ( Dontsum(''), sum(i.num_total[1] for i in payment_items), sum(i.num_total[2] for i in payment_items) ) payment_cat_obj.num_canceled = ( Dontsum(''), sum(i.num_canceled[1] for i in payment_items), sum(i.num_canceled[2] for i in payment_items) ) payment_cat_obj.num_refunded = ( Dontsum(''), sum(i.num_refunded[1] for i in payment_items), sum(i.num_refunded[2] for i in payment_items) ) payment_cat_obj.num_expired = ( Dontsum(''), sum(i.num_expired[1] for i in payment_items), sum(i.num_expired[2] for i in payment_items) ) payment_cat_obj.num_pending = ( Dontsum(''), sum(i.num_pending[1] for i in payment_items), sum(i.num_pending[2] for i in payment_items) ) payment_cat_obj.num_paid = ( Dontsum(''), sum(i.num_paid[1] for i in payment_items), sum(i.num_paid[2] for i in payment_items) ) payment_cat = (payment_cat_obj, payment_items) items_by_category.append(payment_cat) total = { 'num_total': tuplesum(c.num_total for c, i in items_by_category), 'num_pending': tuplesum(c.num_pending for c, i in items_by_category), 'num_expired': tuplesum(c.num_expired for c, i in items_by_category), 'num_canceled': tuplesum(c.num_canceled for c, i in items_by_category), 'num_refunded': tuplesum(c.num_refunded for c, i in items_by_category), 'num_paid': tuplesum(c.num_paid for c, i in items_by_category) } return items_by_category, total