mirror of
https://github.com/pretix/pretix.git
synced 2026-05-04 15:04:03 +00:00
352 lines
13 KiB
Python
352 lines
13 KiB
Python
from datetime import timedelta
|
|
from itertools import groupby
|
|
|
|
from django import forms
|
|
from django.contrib import messages
|
|
from django.core.urlresolvers import reverse
|
|
from django.db.models import Q
|
|
from django.http import HttpResponse
|
|
from django.shortcuts import redirect, render
|
|
from django.utils.functional import cached_property
|
|
from django.utils.timezone import now
|
|
from django.utils.translation import ugettext_lazy as _
|
|
from django.views.generic import DetailView, ListView, TemplateView, View
|
|
|
|
from pretix.base.models import CachedFile, CachedTicket, Item, Order, Quota
|
|
from pretix.base.services.export import export
|
|
from pretix.base.services.orders import mark_order_paid
|
|
from pretix.base.services.stats import order_overview
|
|
from pretix.base.services.tickets import generate
|
|
from pretix.base.signals import (
|
|
register_data_exporters, register_payment_providers,
|
|
register_ticket_outputs,
|
|
)
|
|
from pretix.control.forms.orders import ExtendForm
|
|
from pretix.control.permissions import EventPermissionRequiredMixin
|
|
|
|
|
|
class OrderList(EventPermissionRequiredMixin, ListView):
|
|
model = Order
|
|
context_object_name = 'orders'
|
|
template_name = 'pretixcontrol/orders/index.html'
|
|
paginate_by = 30
|
|
permission = 'can_view_orders'
|
|
|
|
def get_queryset(self):
|
|
qs = Order.objects.current.filter(
|
|
event=self.request.event
|
|
)
|
|
if self.request.GET.get("user", "") != "":
|
|
u = self.request.GET.get("user", "")
|
|
qs = qs.filter(
|
|
Q(user__email__icontains=u) | Q(user__givenname__icontains=u) | Q(user__familyname__icontains=u)
|
|
)
|
|
if self.request.GET.get("status", "") != "":
|
|
s = self.request.GET.get("status", "")
|
|
qs = qs.filter(status=s)
|
|
if self.request.GET.get("item", "") != "":
|
|
i = self.request.GET.get("item", "")
|
|
qs = qs.filter(positions__item_id__in=(i,)).distinct()
|
|
return qs.select_related("user")
|
|
|
|
def get_context_data(self, **kwargs):
|
|
ctx = super().get_context_data(**kwargs)
|
|
ctx['items'] = Item.objects.current.filter(event=self.request.event)
|
|
return ctx
|
|
|
|
|
|
class OrderView(EventPermissionRequiredMixin, DetailView):
|
|
context_object_name = 'order'
|
|
model = Order
|
|
|
|
def get_object(self, queryset=None):
|
|
return Order.objects.current.get(
|
|
event=self.request.event,
|
|
code=self.kwargs['code'].upper()
|
|
)
|
|
|
|
@cached_property
|
|
def order(self):
|
|
return self.get_object()
|
|
|
|
@cached_property
|
|
def payment_provider(self):
|
|
responses = register_payment_providers.send(self.request.event)
|
|
for receiver, response in responses:
|
|
provider = response(self.request.event)
|
|
if provider.identifier == self.order.payment_provider:
|
|
return provider
|
|
|
|
def get_order_url(self):
|
|
return reverse('control:event.order', kwargs={
|
|
'event': self.request.event.slug,
|
|
'organizer': self.request.event.organizer.slug,
|
|
'code': self.order.code
|
|
})
|
|
|
|
|
|
class OrderDetail(OrderView):
|
|
template_name = 'pretixcontrol/order/index.html'
|
|
permission = 'can_view_orders'
|
|
|
|
@cached_property
|
|
def download_buttons(self):
|
|
buttons = []
|
|
responses = register_ticket_outputs.send(self.request.event)
|
|
for receiver, response in responses:
|
|
provider = response(self.request.event)
|
|
if not provider.is_enabled:
|
|
continue
|
|
buttons.append({
|
|
'icon': provider.download_button_icon or 'fa-download',
|
|
'text': provider.download_button_text or 'fa-download',
|
|
'identifier': provider.identifier,
|
|
})
|
|
return buttons
|
|
|
|
def get_context_data(self, **kwargs):
|
|
ctx = super().get_context_data(**kwargs)
|
|
ctx['items'] = self.get_items()
|
|
ctx['event'] = self.request.event
|
|
ctx['download_buttons'] = self.download_buttons
|
|
ctx['can_download'] = (
|
|
self.request.event.settings.ticket_download
|
|
and self.order.status == Order.STATUS_PAID
|
|
)
|
|
ctx['payment'] = self.payment_provider.order_control_render(self.request, self.object)
|
|
return ctx
|
|
|
|
def get_items(self):
|
|
queryset = self.object.positions.all()
|
|
|
|
cartpos = queryset.order_by(
|
|
'item', 'variation'
|
|
).select_related(
|
|
'item', 'variation'
|
|
).prefetch_related(
|
|
'variation__values', 'variation__values__prop', 'item__questions',
|
|
'answers'
|
|
)
|
|
|
|
# Group items of the same variation
|
|
# We do this by list manipulations instead of a GROUP BY query, as
|
|
# Django is unable to join related models in a .values() query
|
|
def keyfunc(pos):
|
|
if (pos.item.admission and self.request.event.settings.attendee_names_asked) \
|
|
or pos.item.questions.all():
|
|
return pos.id, "", "", ""
|
|
return "", pos.item_id, pos.variation_id, pos.price
|
|
|
|
positions = []
|
|
for k, g in groupby(sorted(list(cartpos), key=keyfunc), key=keyfunc):
|
|
g = list(g)
|
|
group = g[0]
|
|
group.count = len(g)
|
|
group.total = group.count * group.price
|
|
group.has_questions = k[0] != ""
|
|
group.cache_answers()
|
|
positions.append(group)
|
|
|
|
return {
|
|
'positions': positions,
|
|
'raw': cartpos,
|
|
'total': self.object.total,
|
|
'payment_fee': self.object.payment_fee,
|
|
}
|
|
|
|
|
|
class OrderTransition(OrderView):
|
|
permission = 'can_change_orders'
|
|
|
|
def post(self, *args, **kwargs):
|
|
to = self.request.POST.get('status', '')
|
|
if self.order.status == 'n' and to == 'p':
|
|
try:
|
|
mark_order_paid(self.order, manual=True)
|
|
except Quota.QuotaExceededException as e:
|
|
messages.error(self.request, str(e))
|
|
else:
|
|
messages.success(self.request, _('The order has been marked as paid.'))
|
|
elif self.order.status == 'n' and to == 'c':
|
|
order = self.order.clone()
|
|
order.status = Order.STATUS_CANCELLED
|
|
order.save()
|
|
messages.success(self.request, _('The order has been cancelled.'))
|
|
elif self.order.status == 'p' and to == 'n':
|
|
order = self.order.clone()
|
|
order.status = Order.STATUS_PENDING
|
|
order.payment_manual = True
|
|
order.save()
|
|
messages.success(self.request, _('The order has been marked as not paid.'))
|
|
elif self.order.status == 'p' and to == 'r':
|
|
ret = self.payment_provider.order_control_refund_perform(self.request, self.order)
|
|
if ret:
|
|
return redirect(ret)
|
|
return redirect(self.get_order_url())
|
|
|
|
def get(self, *args, **kwargs):
|
|
to = self.request.GET.get('status', '')
|
|
if self.order.status == 'n' and to == 'c':
|
|
return render(self.request, 'pretixcontrol/order/cancel.html', {
|
|
'order': self.order,
|
|
})
|
|
elif self.order.status == 'p' and to == 'r':
|
|
return render(self.request, 'pretixcontrol/order/refund.html', {
|
|
'order': self.order,
|
|
'payment': self.payment_provider.order_control_refund_render(self.order),
|
|
})
|
|
else:
|
|
return HttpResponse(status=405)
|
|
|
|
|
|
class OrderDownload(OrderView):
|
|
|
|
@cached_property
|
|
def output(self):
|
|
responses = register_ticket_outputs.send(self.request.event)
|
|
for receiver, response in responses:
|
|
provider = response(self.request.event)
|
|
if provider.identifier == self.kwargs.get('output'):
|
|
return provider
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
if not self.output or not self.output.is_enabled:
|
|
messages.error(request, _('You requested an invalid ticket output type.'))
|
|
return redirect(self.get_order_url())
|
|
if self.order.status != Order.STATUS_PAID:
|
|
messages.error(request, _('Order is not paid.'))
|
|
return redirect(self.get_order_url())
|
|
|
|
try:
|
|
ct = CachedTicket.objects.get(order=self.order, provider=self.output.identifier)
|
|
except CachedTicket.DoesNotExist:
|
|
ct = CachedTicket(order=self.order, provider=self.output.identifier)
|
|
try:
|
|
ct.cachedfile
|
|
except CachedFile.DoesNotExist:
|
|
cf = CachedFile()
|
|
cf.date = now()
|
|
cf.expires = self.request.event.date_from + timedelta(days=30)
|
|
cf.save()
|
|
ct.cachedfile = cf
|
|
ct.save()
|
|
generate(self.order.identity, self.output.identifier)
|
|
return redirect(reverse('cachedfile.download', kwargs={'id': ct.cachedfile.id}))
|
|
|
|
|
|
class OrderExtend(OrderView):
|
|
permission = 'can_change_orders'
|
|
|
|
def post(self, *args, **kwargs):
|
|
if self.order.status != Order.STATUS_PENDING:
|
|
messages.error(self.request, _('This action is only allowed for pending orders.'))
|
|
return self._redirect_back()
|
|
oldvalue = self.order.expires
|
|
|
|
if self.form.is_valid():
|
|
if oldvalue > now():
|
|
self.form.save()
|
|
else:
|
|
is_available = self.order._is_still_available()
|
|
if is_available is True:
|
|
self.form.save()
|
|
messages.success(self.request, _('The payment term has been changed.'))
|
|
else:
|
|
messages.error(self.request, is_available)
|
|
return self._redirect_back()
|
|
else:
|
|
return self.get(*args, **kwargs)
|
|
|
|
def _redirect_back(self):
|
|
return redirect('control:event.order',
|
|
event=self.request.event.slug,
|
|
organizer=self.request.event.organizer.slug,
|
|
code=self.order.code)
|
|
|
|
def get(self, *args, **kwargs):
|
|
if self.order.status != Order.STATUS_PENDING:
|
|
messages.error(self.request, _('This action is only allowed for pending orders.'))
|
|
return self._redirect_back()
|
|
return render(self.request, 'pretixcontrol/order/extend.html', {
|
|
'order': self.order,
|
|
'form': self.form,
|
|
})
|
|
|
|
@cached_property
|
|
def form(self):
|
|
return ExtendForm(instance=self.order,
|
|
data=self.request.POST if self.request.method == "POST" else None)
|
|
|
|
|
|
class OverView(EventPermissionRequiredMixin, TemplateView):
|
|
template_name = 'pretixcontrol/orders/overview.html'
|
|
permission = 'can_view_orders'
|
|
|
|
def get_context_data(self, **kwargs):
|
|
ctx = super().get_context_data()
|
|
ctx['items_by_category'], ctx['total'] = order_overview(self.request.event)
|
|
return ctx
|
|
|
|
|
|
class OrderGo(EventPermissionRequiredMixin, View):
|
|
permission = 'can_view_orders'
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
code = request.GET.get("code")
|
|
try:
|
|
if code.startswith(request.event.slug):
|
|
code = code[len(request.event.slug):]
|
|
order = Order.objects.current.get(code=code, event=request.event)
|
|
return redirect('control:event.order', event=request.event.slug, organizer=request.event.organizer.slug,
|
|
code=order.code)
|
|
except Order.DoesNotExist:
|
|
messages.error(request, _('There is no order with the given order code.'))
|
|
return redirect('control:event.orders', event=request.event.slug, organizer=request.event.organizer.slug)
|
|
|
|
|
|
class ExportView(EventPermissionRequiredMixin, TemplateView):
|
|
permission = 'can_view_orders'
|
|
template_name = 'pretixcontrol/orders/export.html'
|
|
|
|
@cached_property
|
|
def exporters(self):
|
|
exporters = []
|
|
responses = register_data_exporters.send(self.request.event)
|
|
for receiver, response in responses:
|
|
ex = response(self.request.event)
|
|
ex.form = forms.Form(
|
|
data=(self.request.POST if self.request.method == 'POST' else None)
|
|
)
|
|
ex.form.fields = ex.export_form_fields
|
|
exporters.append(ex)
|
|
return exporters
|
|
|
|
def get_context_data(self, **kwargs):
|
|
ctx = super().get_context_data(**kwargs)
|
|
ctx['exporters'] = self.exporters
|
|
return ctx
|
|
|
|
@cached_property
|
|
def exporter(self):
|
|
for ex in self.exporters:
|
|
if ex.identifier == self.request.POST.get("exporter"):
|
|
return ex
|
|
|
|
def post(self, *args, **kwargs):
|
|
if not self.exporter:
|
|
messages.error(self.request, _('The selected exporter was not found.'))
|
|
return redirect('control:event.orders.export', kwargs={
|
|
'event': self.request.event.slug,
|
|
'organizer': self.request.event.organizer.slug
|
|
})
|
|
if not self.exporter.form.is_valid():
|
|
messages.error(self.request, _('There was a problem processing your input. See below for error details.'))
|
|
return self.get(*args, **kwargs)
|
|
|
|
cf = CachedFile()
|
|
cf.date = now()
|
|
cf.expires = now() + timedelta(days=3)
|
|
cf.save()
|
|
export(self.request.event.id, str(cf.id), self.exporter.identifier, self.exporter.form.cleaned_data)
|
|
return redirect(reverse('presale:cachedfile.download', kwargs={'id': cf.id}))
|