symposion_app/vendor/registripe/registripe/views.py
2017-09-30 03:02:40 +10:00

209 lines
5.6 KiB
Python

from registripe import forms
from registripe import models
from django.core.exceptions import ValidationError
from django.conf import settings
from django.contrib import messages
from django.contrib.auth.decorators import user_passes_test
from django.db import transaction
from django.http import Http404
from django.http import HttpResponse
from django.shortcuts import redirect, render
from registrasion.controllers.credit_note import CreditNoteController
from registrasion.controllers.invoice import InvoiceController
from pinax.stripe import actions
from pinax.stripe.actions import refunds as pinax_stripe_actions_refunds
from registrasion.models import commerce
from stripe.error import StripeError
from symposion.conference.models import Conference
CURRENCY = settings.INVOICE_CURRENCY
CONFERENCE_ID = settings.CONFERENCE_ID
def _staff_only(user):
''' Returns true if the user is staff. '''
return user.is_staff
def form_handler(request):
''' Renders a js file to process Stripe payments. '''
data = {
"PINAX_STRIPE_PUBLIC_KEY": settings.PINAX_STRIPE_PUBLIC_KEY,
}
return render(
request,
"registrasion/stripe/js/form_handler.js",
data,
content_type="text/javascript",
)
script_template = "Stripe.setPublishableKey('%s');"
script = script_template % settings.PINAX_STRIPE_PUBLIC_KEY
return HttpResponse(script, content_type="text/javascript")
def card(request, invoice_id, access_code=None):
''' View that shows and processes a Stripe CreditCardForm to pay the given
invoice. Redirects back to the invoice once the invoice is fully paid.
Arguments:
invoice_id (castable to str): The invoice id for the invoice to pay.
access_code (str): The optional access code for the invoice (for
unauthenticated payment)
'''
form = forms.CreditCardForm(request.POST or None)
inv = InvoiceController.for_id_or_404(str(invoice_id))
if not inv.can_view(user=request.user, access_code=access_code):
raise Http404()
args = [inv.invoice.id]
if access_code:
args.append(access_code)
to_invoice = redirect("invoice", *args)
if inv.invoice.balance_due() <= 0:
return to_invoice
if request.POST and form.is_valid():
try:
inv.validate_allowed_to_pay() # Verify that we're allowed to do this.
process_card(request, form, inv)
return to_invoice
except StripeError as e:
form.add_error(None, ValidationError(e))
except ValidationError as ve:
form.add_error(None, ve)
data = {
"invoice": inv.invoice,
"form": form,
}
return render(
request, "registrasion/stripe/credit_card_payment.html", data
)
@transaction.atomic
def process_card(request, form, inv):
''' Processes the given credit card form
Arguments:
request: the current request context
form: a CreditCardForm
inv: an InvoiceController
'''
conference = Conference.objects.get(id=CONFERENCE_ID)
amount_to_pay = inv.invoice.balance_due()
user = inv.invoice.user
token = form.cleaned_data["stripe_token"]
customer = actions.customers.get_customer_for_user(user)
if not customer:
customer = actions.customers.create(user)
card = actions.sources.create_card(customer, token)
description = "Payment for %s invoice #%s" % (
conference.title, inv.invoice.id
)
charge = actions.charges.create(
amount_to_pay,
customer,
source=card,
currency=CURRENCY,
description=description,
capture=True,
)
receipt = charge.stripe_charge.id
reference = "Paid with Stripe reference: " + receipt
# Create the payment object
models.StripePayment.objects.create(
invoice=inv.invoice,
reference=reference,
amount=charge.amount,
charge=charge,
)
inv.update_status()
messages.success(request, "This invoice was successfully paid.")
@user_passes_test(_staff_only)
def refund(request, credit_note_id):
''' Allows staff to select a Stripe charge for the owner of the credit
note, and refund the credit note into stripe. '''
cn = CreditNoteController.for_id_or_404(str(credit_note_id))
to_credit_note = redirect("credit_note", cn.credit_note.id)
if not cn.credit_note.is_unclaimed:
return to_credit_note
form = forms.StripeRefundForm(
request.POST or None,
user=cn.credit_note.invoice.user,
min_value=cn.credit_note.value,
)
if request.POST and form.is_valid():
try:
process_refund(cn, form)
return to_credit_note
except StripeError as se:
form.add_error(None, ValidationError(se))
data = {
"credit_note": cn.credit_note,
"form": form,
}
return render(
request, "registrasion/stripe/refund.html", data
)
def process_refund(cn, form):
payment = form.cleaned_data["payment"]
charge = payment.charge
to_refund = cn.credit_note.value
stripe_charge_id = charge.stripe_charge.id
# Test that the given charge is allowed to be refunded.
max_refund = actions.charges.calculate_refund_amount(charge)
if max_refund < to_refund:
raise ValidationError(
"You must select a payment holding greater value than "
"the credit note."
)
refund = actions.refunds.create(charge, to_refund)
models.StripeCreditNoteRefund.objects.create(
parent=cn.credit_note,
charge=charge,
reference="Refunded %s to Stripe charge %s" % (
to_refund, stripe_charge_id
)
)