bornhack-website/src/shop/views.py

683 lines
23 KiB
Python
Raw Normal View History

import logging
from collections import OrderedDict
from django.conf import settings
2016-05-12 17:08:54 +00:00
from django.contrib import messages
2016-05-15 22:09:00 +00:00
from django.contrib.auth.mixins import LoginRequiredMixin
from django.db.models import Count
2018-03-04 14:38:40 +00:00
from django.http import (
Http404,
2018-03-04 14:38:40 +00:00
HttpResponse,
HttpResponseBadRequest,
HttpResponseRedirect,
2018-03-04 14:38:40 +00:00
)
2016-05-16 19:54:25 +00:00
from django.shortcuts import get_object_or_404
from django.urls import reverse, reverse_lazy
from django.utils import timezone
from django.utils.decorators import method_decorator
from django.views.decorators.csrf import csrf_exempt
from django.views.generic import DetailView, FormView, ListView, View
2016-05-29 10:10:51 +00:00
from django.views.generic.base import RedirectView
from django.views.generic.detail import SingleObjectMixin
2016-05-15 22:09:00 +00:00
from shop.models import (
CreditNote,
EpayCallback,
EpayPayment,
2016-05-15 22:09:00 +00:00
Order,
OrderProductRelation,
Product,
2016-05-15 22:09:00 +00:00
ProductCategory,
)
from vendor.coinify.coinify_callback import CoinifyCallback
2018-03-04 14:38:40 +00:00
from .coinify import (
create_coinify_invoice,
2019-03-29 21:06:52 +00:00
process_coinify_invoice_json,
save_coinify_callback,
2018-03-04 14:38:40 +00:00
)
from .epay import calculate_epay_hash, validate_epay_callback
from .forms import OrderProductRelationForm, OrderProductRelationFormSet
2017-03-23 17:32:13 +00:00
logger = logging.getLogger("bornhack.%s" % __name__)
2018-03-04 14:38:40 +00:00
# Mixins
2016-06-19 19:37:25 +00:00
class EnsureCreditNoteHasPDFMixin(SingleObjectMixin):
model = CreditNote
def dispatch(self, request, *args, **kwargs):
if not self.get_object().pdf:
messages.error(request, "This creditnote has no PDF yet!")
2019-03-29 21:06:52 +00:00
return HttpResponseRedirect(reverse_lazy("shop:creditnote_list"))
2016-06-19 19:37:25 +00:00
return super(EnsureCreditNoteHasPDFMixin, self).dispatch(
request, *args, **kwargs
)
class EnsureUserOwnsCreditNoteMixin(SingleObjectMixin):
model = CreditNote
def dispatch(self, request, *args, **kwargs):
# If the user does not own this creditnote OR is not staff
if not request.user.is_staff:
if self.get_object().user != request.user:
raise Http404("CreditNote not found")
return super(EnsureUserOwnsCreditNoteMixin, self).dispatch(
request, *args, **kwargs
)
class EnsureUserOwnsOrderMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
2016-06-19 19:37:25 +00:00
# If the user does not own this order OR is not staff
2016-05-31 21:30:18 +00:00
if not request.user.is_staff:
if self.get_object().user != request.user:
raise Http404("Order not found")
2019-03-29 21:06:52 +00:00
return super(EnsureUserOwnsOrderMixin, self).dispatch(request, *args, **kwargs)
class EnsureUnpaidOrderMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
if self.get_object().paid:
messages.error(request, "This order is already paid for!")
2018-03-04 14:38:40 +00:00
return HttpResponseRedirect(
2019-03-29 21:06:52 +00:00
reverse_lazy("shop:order_detail", kwargs={"pk": self.get_object().pk})
2018-03-04 14:38:40 +00:00
)
2019-03-29 21:06:52 +00:00
return super(EnsureUnpaidOrderMixin, self).dispatch(request, *args, **kwargs)
class EnsurePaidOrderMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
if not self.get_object().paid:
messages.error(request, "This order is not paid for!")
2018-03-04 14:38:40 +00:00
return HttpResponseRedirect(
2019-03-29 21:06:52 +00:00
reverse_lazy("shop:order_detail", kwargs={"pk": self.get_object().pk})
2018-03-04 14:38:40 +00:00
)
2019-03-29 21:06:52 +00:00
return super(EnsurePaidOrderMixin, self).dispatch(request, *args, **kwargs)
2016-05-17 06:34:54 +00:00
class EnsureClosedOrderMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
if self.get_object().open is not None:
2019-03-29 21:06:52 +00:00
messages.error(request, "This order is still open!")
2018-03-04 14:38:40 +00:00
return HttpResponseRedirect(
2019-03-29 21:06:52 +00:00
reverse_lazy("shop:order_detail", kwargs={"pk": self.get_object().pk})
2018-03-04 14:38:40 +00:00
)
2016-05-17 06:34:54 +00:00
2019-03-29 21:06:52 +00:00
return super(EnsureClosedOrderMixin, self).dispatch(request, *args, **kwargs)
2016-05-17 06:34:54 +00:00
class EnsureOrderHasProductsMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
if not self.get_object().products.count() > 0:
2019-03-29 21:06:52 +00:00
messages.error(request, "This order has no products!")
return HttpResponseRedirect(reverse_lazy("shop:index"))
2016-05-17 06:34:54 +00:00
return super(EnsureOrderHasProductsMixin, self).dispatch(
request, *args, **kwargs
)
2016-05-31 19:19:31 +00:00
2016-06-01 09:10:06 +00:00
class EnsureOrderIsNotCancelledMixin(SingleObjectMixin):
model = Order
def dispatch(self, request, *args, **kwargs):
if self.get_object().cancelled:
messages.error(
2019-03-29 21:06:52 +00:00
request, "Order #{} is cancelled!".format(self.get_object().id)
2016-06-01 09:10:06 +00:00
)
2019-03-29 21:06:52 +00:00
return HttpResponseRedirect(reverse_lazy("shop:index"))
2016-06-01 09:10:06 +00:00
2016-06-01 09:11:24 +00:00
return super(EnsureOrderIsNotCancelledMixin, self).dispatch(
2016-06-01 09:10:06 +00:00
request, *args, **kwargs
)
2018-03-04 14:38:40 +00:00
# Shop views
2016-05-12 17:08:54 +00:00
class ShopIndexView(ListView):
model = Product
template_name = "shop_index.html"
2019-03-29 21:06:52 +00:00
context_object_name = "products"
def get_queryset(self):
queryset = super(ShopIndexView, self).get_queryset()
2020-02-07 17:46:34 +00:00
return queryset.available().order_by(
"category__weight", "category__name", "price", "name"
)
2016-05-13 06:36:56 +00:00
def get_context_data(self, **kwargs):
2016-05-13 06:50:04 +00:00
context = super(ShopIndexView, self).get_context_data(**kwargs)
2019-03-29 21:06:52 +00:00
if "category" in self.request.GET:
category = self.request.GET.get("category")
2016-05-17 19:55:09 +00:00
# is this a public category
2016-05-17 19:24:04 +00:00
try:
2016-05-17 19:42:28 +00:00
categoryobj = ProductCategory.objects.get(slug=category)
2016-05-17 19:24:04 +00:00
if not categoryobj.public:
raise Http404("Category not found")
except ProductCategory.DoesNotExist:
2016-05-17 19:10:01 +00:00
raise Http404("Category not found")
2016-05-17 19:55:09 +00:00
# filter products by the chosen category
2019-03-29 21:06:52 +00:00
context["products"] = context["products"].filter(category__slug=category)
context["current_category"] = categoryobj
context["categories"] = ProductCategory.objects.annotate(
num_products=Count("products")
2016-05-15 22:09:00 +00:00
).filter(
2016-05-17 19:10:01 +00:00
num_products__gt=0,
public=True,
2019-03-29 21:06:52 +00:00
products__available_in__contains=timezone.now(),
2016-05-15 22:09:00 +00:00
)
return context
2016-05-13 06:36:56 +00:00
class ProductDetailView(FormView, DetailView):
2016-05-15 22:09:00 +00:00
model = Product
2019-03-29 21:06:52 +00:00
template_name = "product_detail.html"
form_class = OrderProductRelationForm
2019-03-29 21:06:52 +00:00
context_object_name = "product"
def get_form_kwargs(self):
kwargs = super().get_form_kwargs()
2019-03-29 21:06:52 +00:00
if hasattr(self, "opr"):
kwargs["instance"] = self.opr
return kwargs
def get_initial(self):
2019-03-29 21:06:52 +00:00
if hasattr(self, "opr"):
return {"quantity": self.opr.quantity}
return None
def get_context_data(self, **kwargs):
# If the OrderProductRelation already exists it has a primary key in the database
if self.request.user.is_authenticated and self.opr.pk:
2019-03-29 21:06:52 +00:00
kwargs["already_in_order"] = True
return super().get_context_data(**kwargs)
2016-05-17 13:09:31 +00:00
def dispatch(self, request, *args, **kwargs):
self.object = self.get_object()
if not self.object.category.public:
# this product is not publicly available
raise Http404("Product not found")
if self.request.user.is_authenticated:
try:
self.opr = OrderProductRelation.objects.get(
order__user=self.request.user,
order__open__isnull=False,
2019-03-29 21:06:52 +00:00
product=self.object,
)
except OrderProductRelation.DoesNotExist:
2019-03-29 21:06:52 +00:00
self.opr = OrderProductRelation(product=self.get_object(), quantity=1)
2019-03-29 21:06:52 +00:00
return super(ProductDetailView, self).dispatch(request, *args, **kwargs)
2016-05-17 18:42:02 +00:00
2016-05-12 17:08:54 +00:00
def form_valid(self, form):
opr = form.save(commit=False)
2016-05-15 22:09:00 +00:00
if not opr.pk:
2019-03-29 21:06:52 +00:00
opr.order, _ = Order.objects.get_or_create(
user=self.request.user, open=True, cancelled=False
)
2016-05-15 22:09:00 +00:00
opr.save()
2016-05-15 22:09:00 +00:00
messages.info(
self.request,
2019-03-29 21:06:52 +00:00
"{}x {} has been added to your order.".format(
opr.quantity, opr.product.name
),
2016-05-15 22:09:00 +00:00
)
2016-05-12 17:08:54 +00:00
2016-05-15 22:09:00 +00:00
# done
return super(ProductDetailView, self).form_valid(form)
2016-05-12 17:08:54 +00:00
def get_success_url(self):
return reverse("shop:index")
2016-05-12 17:08:54 +00:00
class OrderListView(LoginRequiredMixin, ListView):
model = Order
template_name = "order_list.html"
2019-03-29 21:06:52 +00:00
context_object_name = "orders"
2016-06-01 09:10:06 +00:00
def get_queryset(self):
queryset = super(OrderListView, self).get_queryset()
return queryset.filter(user=self.request.user).not_cancelled()
2019-03-29 21:06:52 +00:00
class OrderDetailView(
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureOrderHasProductsMixin,
EnsureOrderIsNotCancelledMixin,
DetailView,
):
model = Order
template_name = "order_detail.html"
2019-03-29 21:06:52 +00:00
context_object_name = "order"
def get_context_data(self, **kwargs):
2019-03-29 21:06:52 +00:00
if "order_product_formset" not in kwargs:
kwargs["order_product_formset"] = OrderProductRelationFormSet(
queryset=OrderProductRelation.objects.filter(order=self.get_object())
)
return super().get_context_data(**kwargs)
def get(self, request, *args, **kwargs):
order = self.get_object()
if order.open is None and not order.paid:
return HttpResponseRedirect(
reverse("shop:order_review_and_pay", kwargs={"pk": order.pk})
)
return super().get(request, *args, **kwargs)
def post(self, request, *args, **kwargs):
self.object = self.get_object()
order = self.object
# First check if the user is removing a product from the order.
2019-03-29 21:06:52 +00:00
product_remove = request.POST.get("remove_product")
if product_remove:
order.orderproductrelation_set.filter(pk=product_remove).delete()
if not order.products.count() > 0:
order.mark_as_cancelled()
2019-03-29 21:06:52 +00:00
messages.info(request, "Order cancelled!")
return HttpResponseRedirect(reverse_lazy("shop:index"))
# Then see if the user is cancelling the order.
2019-03-29 21:06:52 +00:00
elif "cancel_order" in request.POST:
order.mark_as_cancelled()
2019-03-29 21:06:52 +00:00
messages.info(request, "Order cancelled!")
return HttpResponseRedirect(reverse_lazy("shop:index"))
# The user is not removing products or cancelling the order,
# so from now on we do stuff that require us to check stock.
# We use a formset for this to be able to display exactly
# which product is not in stock if that is the case.
else:
formset = OrderProductRelationFormSet(
2019-03-29 21:06:52 +00:00
request.POST, queryset=OrderProductRelation.objects.filter(order=order)
)
# If the formset is not valid it means that we cannot fulfill the order, so return and inform the user.
if not formset.is_valid():
messages.error(
request,
2019-03-29 21:06:52 +00:00
"Some of the products you are ordering are out of stock. Review the order and try again.",
2018-03-04 14:38:40 +00:00
)
return self.render_to_response(
self.get_context_data(order_product_formset=formset)
)
# No stock issues, proceed to check if the user is updating or proceeding to review and pay the order.
if "update_order" or "review_and_pay" in request.POST:
# We have already made sure the formset is valid, so just save it to update quantities.
formset.save()
2019-03-29 21:06:52 +00:00
order.customer_comment = request.POST.get("customer_comment") or ""
order.invoice_address = request.POST.get("invoice_address") or ""
order.save()
if "review_and_pay" in request.POST:
return HttpResponseRedirect(
reverse("shop:order_review_and_pay", kwargs={"pk": order.pk})
)
return super(OrderDetailView, self).get(request, *args, **kwargs)
class OrderReviewAndPayView(
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureOrderHasProductsMixin,
EnsureUnpaidOrderMixin,
EnsureOrderIsNotCancelledMixin,
DetailView,
):
template_name = "order_review.html"
context_object_name = "order"
def post(self, request, *args, **kwargs):
self.object = self.get_object()
order = self.object
payment_method = request.POST.get("payment_method")
if payment_method in order.PAYMENT_METHODS:
if not request.POST.get("accept_terms"):
messages.error(
request,
"You need to accept the general terms and conditions before you can continue!",
)
return self.render_to_response(self.get_context_data())
# Set payment method and mark the order as closed
order.payment_method = payment_method
order.open = None
order.save()
reverses = {
Order.CREDIT_CARD: reverse_lazy(
"shop:epay_form", kwargs={"pk": order.id}
),
Order.BLOCKCHAIN: reverse_lazy(
"shop:coinify_pay", kwargs={"pk": order.id}
),
Order.BANK_TRANSFER: reverse_lazy(
"shop:bank_transfer", kwargs={"pk": order.id}
),
Order.IN_PERSON: reverse_lazy(
"shop:in_person", kwargs={"pk": order.id}
),
}
return HttpResponseRedirect(reverses[payment_method])
2019-03-29 21:06:52 +00:00
class DownloadInvoiceView(
LoginRequiredMixin, EnsureUserOwnsOrderMixin, SingleObjectMixin, View
2019-03-29 21:06:52 +00:00
):
model = Order
def get(self, request, *args, **kwargs):
2019-07-09 08:38:14 +00:00
"""
The file we return is determined by the orders paid status.
If the order is unpaid we return a proforma invoice PDF
If the order is paid we return a normal Invoice PDF
"""
if self.get_object().paid:
pdfobj = self.get_object().invoice
else:
pdfobj = self.get_object()
if not pdfobj.pdf:
messages.error(request, "No PDF has been generated yet!")
return HttpResponseRedirect(
reverse_lazy("shop:order_detail", kwargs={"pk": self.get_object().pk})
)
2019-03-29 21:06:52 +00:00
response = HttpResponse(content_type="application/pdf")
response["Content-Disposition"] = 'attachment; filename="%s"' % pdfobj.filename
2019-07-09 08:38:14 +00:00
response.write(pdfobj.pdf.read())
return response
2016-06-19 19:37:25 +00:00
class CreditNoteListView(LoginRequiredMixin, ListView):
model = CreditNote
template_name = "creditnote_list.html"
2019-03-29 21:06:52 +00:00
context_object_name = "creditnotes"
2016-06-19 19:37:25 +00:00
def get_queryset(self):
queryset = super().get_queryset()
return queryset.filter(user=self.request.user)
2016-06-19 19:37:25 +00:00
2019-03-29 21:06:52 +00:00
class DownloadCreditNoteView(
LoginRequiredMixin,
EnsureUserOwnsCreditNoteMixin,
EnsureCreditNoteHasPDFMixin,
SingleObjectMixin,
View,
):
2016-06-19 19:37:25 +00:00
model = CreditNote
def get(self, request, *args, **kwargs):
2019-03-29 21:06:52 +00:00
response = HttpResponse(content_type="application/pdf")
response["Content-Disposition"] = (
'attachment; filename="%s"' % self.get_object().filename
)
2016-06-19 20:04:24 +00:00
response.write(self.get_object().pdf.read())
2016-06-19 19:37:25 +00:00
return response
2016-11-09 11:28:34 +00:00
class OrderMarkAsPaidView(LoginRequiredMixin, SingleObjectMixin, View):
model = Order
def get(self, request, *args, **kwargs):
if not request.user.is_staff:
2019-03-29 21:06:52 +00:00
messages.error(request, "You do not have permissions to do that.")
return HttpResponseRedirect(reverse_lazy("shop:index"))
2016-11-09 11:28:34 +00:00
else:
2019-03-29 21:06:52 +00:00
messages.success(request, "The order has been marked as paid.")
2016-11-09 11:28:34 +00:00
order = self.get_object()
order.mark_as_paid()
2019-03-29 21:06:52 +00:00
return HttpResponseRedirect(request.META.get("HTTP_REFERER"))
2016-11-09 11:28:34 +00:00
2018-03-04 14:38:40 +00:00
# Epay views
2019-03-29 21:06:52 +00:00
class EpayFormView(
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureUnpaidOrderMixin,
EnsureClosedOrderMixin,
EnsureOrderHasProductsMixin,
DetailView,
):
model = Order
2019-03-29 21:06:52 +00:00
template_name = "epay_form.html"
2016-05-16 13:31:37 +00:00
def get_context_data(self, **kwargs):
order = self.get_object()
2016-05-16 13:31:37 +00:00
context = super(EpayFormView, self).get_context_data(**kwargs)
2019-03-29 21:06:52 +00:00
context["merchant_number"] = settings.EPAY_MERCHANT_NUMBER
context["description"] = order.description
context["amount"] = order.total * 100
context["order_id"] = order.pk
context["accept_url"] = order.get_epay_accept_url(self.request)
context["cancel_url"] = order.get_cancel_url(self.request)
context["callback_url"] = order.get_epay_callback_url(self.request)
context["epay_hash"] = calculate_epay_hash(order, self.request)
2016-05-16 13:31:37 +00:00
return context
class EpayCallbackView(SingleObjectMixin, View):
2016-05-29 15:12:09 +00:00
model = Order
def get(self, request, *args, **kwargs):
2019-03-29 21:06:52 +00:00
callback = EpayCallback.objects.create(payload=request.GET)
2016-05-16 13:31:37 +00:00
2019-03-29 21:06:52 +00:00
if "orderid" in request.GET:
query = OrderedDict(
2019-03-29 21:06:52 +00:00
[tuple(x.split("=")) for x in request.META["QUERY_STRING"].split("&")]
2016-05-16 13:31:37 +00:00
)
2019-03-29 21:06:52 +00:00
order = get_object_or_404(Order, pk=query.get("orderid"))
if order.pk != self.get_object().pk:
2017-03-23 17:32:13 +00:00
logger.error("bad epay callback, orders do not match!")
return HttpResponse(status=400)
2016-05-16 13:31:37 +00:00
if validate_epay_callback(query):
2019-03-29 21:06:52 +00:00
callback.md5valid = True
callback.save()
else:
2017-03-23 17:32:13 +00:00
logger.error("bad epay callback!")
2016-05-16 13:31:37 +00:00
return HttpResponse(status=400)
2017-03-23 17:32:13 +00:00
if order.paid:
2018-03-04 14:38:40 +00:00
# this order is already paid, perhaps we are seeing a double callback?
2019-03-29 21:06:52 +00:00
return HttpResponse("OK")
2018-03-04 14:38:40 +00:00
# epay callback is valid - has the order been paid in full?
2019-03-29 21:06:52 +00:00
if int(query["amount"]) == order.total * 100:
2018-03-04 14:38:40 +00:00
# create an EpayPayment object linking the callback to the order
EpayPayment.objects.create(
2019-03-29 21:06:52 +00:00
order=order, callback=callback, txnid=query.get("txnid")
)
2018-03-04 14:38:40 +00:00
# and mark order as paid (this will create tickets)
2018-01-07 16:36:44 +00:00
order.mark_as_paid(request)
else:
2017-03-23 17:32:13 +00:00
logger.error("valid epay callback with wrong amount detected")
2016-05-16 13:31:37 +00:00
else:
return HttpResponse(status=400)
2019-03-29 21:06:52 +00:00
return HttpResponse("OK")
2016-05-16 14:31:51 +00:00
2019-03-29 21:06:52 +00:00
class EpayThanksView(
LoginRequiredMixin, EnsureUserOwnsOrderMixin, EnsureClosedOrderMixin, DetailView
):
2016-05-16 17:31:14 +00:00
model = Order
2019-03-29 21:06:52 +00:00
template_name = "epay_thanks.html"
2016-05-16 17:31:14 +00:00
def dispatch(self, request, *args, **kwargs):
if request.GET:
# epay redirects the user back to our accepturl with a long
# and ugly querystring, redirect user to the clean url
2018-03-04 14:38:40 +00:00
return HttpResponseRedirect(
2019-03-29 21:06:52 +00:00
reverse("shop:epay_thanks", kwargs={"pk": self.get_object().pk})
2018-03-04 14:38:40 +00:00
)
2019-03-29 21:06:52 +00:00
return super(EpayThanksView, self).dispatch(request, *args, **kwargs)
2016-05-29 14:40:32 +00:00
2016-11-09 11:28:34 +00:00
2018-03-04 14:38:40 +00:00
# Bank Transfer view
2016-05-16 17:31:14 +00:00
2019-03-29 21:06:52 +00:00
class BankTransferView(
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureUnpaidOrderMixin,
EnsureOrderHasProductsMixin,
DetailView,
):
2016-05-16 17:31:14 +00:00
model = Order
2019-03-29 21:06:52 +00:00
template_name = "bank_transfer.html"
2016-05-16 14:31:51 +00:00
2016-06-01 07:29:21 +00:00
def get_context_data(self, **kwargs):
context = super(BankTransferView, self).get_context_data(**kwargs)
2019-03-29 21:06:52 +00:00
context["iban"] = settings.BANKACCOUNT_IBAN
context["swiftbic"] = settings.BANKACCOUNT_SWIFTBIC
context["orderid"] = self.get_object().pk
context["regno"] = settings.BANKACCOUNT_REG
context["accountno"] = settings.BANKACCOUNT_ACCOUNT
context["total"] = self.get_object().total
2016-06-01 07:29:21 +00:00
return context
2016-11-09 11:27:42 +00:00
# In-person (izettle) payment view
2016-11-09 11:27:42 +00:00
2019-03-29 21:06:52 +00:00
class PayInPersonView(
2019-03-29 21:06:52 +00:00
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureUnpaidOrderMixin,
EnsureOrderHasProductsMixin,
DetailView,
):
2016-11-09 11:27:42 +00:00
model = Order
template_name = "in_person.html"
2016-11-09 11:27:42 +00:00
2016-05-25 20:48:02 +00:00
2018-03-04 14:38:40 +00:00
# Coinify views
2016-11-09 11:28:34 +00:00
2019-03-29 21:06:52 +00:00
class CoinifyRedirectView(
LoginRequiredMixin,
EnsureUserOwnsOrderMixin,
EnsureUnpaidOrderMixin,
EnsureClosedOrderMixin,
EnsureOrderHasProductsMixin,
SingleObjectMixin,
RedirectView,
):
2016-05-25 20:48:02 +00:00
model = Order
2016-05-29 10:08:29 +00:00
def dispatch(self, request, *args, **kwargs):
2016-05-25 20:48:02 +00:00
order = self.get_object()
# create a new coinify invoice if needed
if not order.coinifyapiinvoice:
coinifyinvoice = create_coinify_invoice(order, request)
if not coinifyinvoice:
2019-03-29 21:06:52 +00:00
messages.error(
request,
"There was a problem with the payment provider. Please try again later",
)
2018-03-04 14:38:40 +00:00
return HttpResponseRedirect(
2019-03-29 21:06:52 +00:00
reverse_lazy(
"shop:order_detail", kwargs={"pk": self.get_object().pk}
)
2018-03-04 14:38:40 +00:00
)
2019-03-29 21:06:52 +00:00
return super(CoinifyRedirectView, self).dispatch(request, *args, **kwargs)
2016-05-29 10:08:29 +00:00
def get_redirect_url(self, *args, **kwargs):
2019-03-29 21:06:52 +00:00
return self.get_object().coinifyapiinvoice.invoicejson["payment_url"]
2016-05-25 20:48:02 +00:00
2016-05-29 08:43:38 +00:00
class CoinifyCallbackView(SingleObjectMixin, View):
model = Order
@method_decorator(csrf_exempt)
def dispatch(self, *args, **kwargs):
return super(CoinifyCallbackView, self).dispatch(*args, **kwargs)
2016-05-25 20:48:02 +00:00
def post(self, request, *args, **kwargs):
# save callback and parse json payload
2017-07-11 05:02:17 +00:00
callbackobject = save_coinify_callback(request, self.get_object())
# do we have a json body?
if not callbackobject.payload:
# no, return an error
2019-03-29 21:06:52 +00:00
logger.error(
"unable to parse JSON body in callback for order %s"
% callbackobject.order.id
)
return HttpResponseBadRequest("unable to parse json")
# initiate SDK
2019-03-29 21:06:52 +00:00
sdk = CoinifyCallback(settings.COINIFY_IPN_SECRET.encode("utf-8"))
# attemt to validate the callbackc
2019-03-29 21:06:52 +00:00
if sdk.validate_callback(
request.body, request.META["HTTP_X_COINIFY_CALLBACK_SIGNATURE"]
):
# mark callback as valid in db
2018-03-04 14:38:40 +00:00
callbackobject.valid = True
callbackobject.save()
2016-05-25 20:48:02 +00:00
else:
2017-03-23 17:32:13 +00:00
logger.error("invalid coinify callback detected")
2019-03-29 21:06:52 +00:00
return HttpResponseBadRequest("something is fucky")
2016-05-25 20:48:02 +00:00
2019-03-29 21:06:52 +00:00
if (
callbackobject.payload["event"] == "invoice_state_change"
or callbackobject.payload["event"] == "invoice_manual_resend"
):
2018-03-04 14:38:40 +00:00
process_coinify_invoice_json(
2019-03-29 21:06:52 +00:00
invoicejson=callbackobject.payload["data"],
order=self.get_object(),
request=request,
2018-03-04 14:38:40 +00:00
)
2019-03-29 21:06:52 +00:00
return HttpResponse("OK")
else:
2019-03-29 21:06:52 +00:00
logger.error(
"unsupported callback event %s" % callbackobject.payload["event"]
)
return HttpResponseBadRequest("unsupported event")
2016-05-29 08:43:38 +00:00
2019-03-29 21:06:52 +00:00
class CoinifyThanksView(
LoginRequiredMixin, EnsureUserOwnsOrderMixin, EnsureClosedOrderMixin, DetailView
):
2016-05-29 08:43:38 +00:00
model = Order
2019-03-29 21:06:52 +00:00
template_name = "coinify_thanks.html"