Files
pretix_original/src/pretix/plugins/banktransfer/views.py
2023-09-08 14:11:45 +02:00

934 lines
38 KiB
Python

#
# This file is part of pretix (Community Edition).
#
# Copyright (C) 2014-2020 Raphael Michel and contributors
# Copyright (C) 2020-2021 rami.io GmbH and contributors
#
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
# Public License as published by the Free Software Foundation in version 3 of the License.
#
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
# this file, see <https://pretix.eu/about/en/license>.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
# details.
#
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
# <https://www.gnu.org/licenses/>.
#
# This file is based on an earlier version of pretix which was released under the Apache License 2.0. The full text of
# the Apache License 2.0 can be obtained at <http://www.apache.org/licenses/LICENSE-2.0>.
#
# This file may have since been changed and any changes are released under the terms of AGPLv3 as described above. A
# full history of changes and contributors is available at <https://github.com/pretix/pretix>.
#
# This file contains Apache-licensed contributions copyrighted by: Brandon, Flavia Bastos, Mohit Jindal, Tobias Kunze,
# alice
#
# Unless required by applicable law or agreed to in writing, software distributed under the Apache License 2.0 is
# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations under the License.
import csv
import itertools
import json
import logging
from collections import defaultdict
from datetime import timedelta
from decimal import Decimal
from typing import Set
from django import forms
from django.contrib import messages
from django.core.exceptions import ValidationError
from django.core.validators import validate_email
from django.db import transaction
from django.db.models import Count, Q, QuerySet
from django.http import FileResponse, Http404, JsonResponse
from django.shortcuts import get_object_or_404, redirect, render
from django.urls import reverse
from django.utils.decorators import method_decorator
from django.utils.functional import cached_property
from django.utils.timezone import now
from django.utils.translation import gettext as _
from django.views.decorators.clickjacking import xframe_options_exempt
from django.views.generic import DetailView, FormView, ListView, View
from django.views.generic.detail import SingleObjectMixin
from localflavor.generic.forms import BICFormField, IBANFormField
from pretix.base.forms.widgets import DatePickerWidget
from pretix.base.models import Event, Order, OrderPayment, OrderRefund, Quota
from pretix.base.services.mail import SendMailException
from pretix.base.settings import SettingsSandbox
from pretix.base.templatetags.money import money_filter
from pretix.control.permissions import (
EventPermissionRequiredMixin, OrganizerPermissionRequiredMixin,
)
from pretix.control.views.organizer import OrganizerDetailViewMixin
from pretix.helpers.json import CustomJSONEncoder
from pretix.plugins.banktransfer import csvimport, mt940import
from pretix.plugins.banktransfer.models import (
BankImportJob, BankTransaction, RefundExport,
)
from pretix.plugins.banktransfer.payment import BankTransfer
from pretix.plugins.banktransfer.refund_export import (
build_sepa_xml, get_refund_export_csv,
)
from pretix.plugins.banktransfer.tasks import process_banktransfers
from pretix.presale.views import EventViewMixin
from pretix.presale.views.order import OrderDetailMixin
logger = logging.getLogger('pretix.plugins.banktransfer')
class ActionView(View):
permission = 'can_change_orders'
def _discard(self, trans):
trans.state = BankTransaction.STATE_DISCARDED
trans.shred_private_data()
trans.save()
return JsonResponse({
'status': 'ok'
})
def _retry(self, trans):
return self._accept_ignore_amount(trans)
def _accept_ignore_amount(self, trans):
if trans.currency and trans.order and trans.currency != trans.order.event.currency:
return JsonResponse({
'status': 'error',
'message': _('Currencies do not match.')
})
if trans.amount < Decimal('0.00'):
ref = trans.order.refunds.filter(
amount=trans.amount * -1,
provider='manual',
state__in=(OrderRefund.REFUND_STATE_CREATED, OrderRefund.REFUND_STATE_CREATED)
).first()
p = trans.order.payments.filter(
amount=trans.amount * -1,
provider='banktransfer',
state__in=(OrderPayment.PAYMENT_STATE_CONFIRMED, OrderPayment.PAYMENT_STATE_REFUNDED)
).first()
if ref:
ref.done(user=self.request.user)
trans.state = BankTransaction.STATE_VALID
trans.save()
return JsonResponse({
'status': 'ok',
})
elif p:
p.create_external_refund(
amount=trans.amount * -1,
info=json.dumps({
'reference': trans.reference,
'date': trans.date,
'payer': trans.payer,
'iban': trans.iban,
'bic': trans.bic,
'trans_id': trans.pk
})
)
trans.state = BankTransaction.STATE_VALID
trans.save()
return JsonResponse({
'status': 'ok',
})
else:
return JsonResponse({
'status': 'error',
'message': _('Negative amount but refund can\'t be logged, please create manual refund first.')
})
p = trans.order.payments.get_or_create(
amount=trans.amount,
provider='banktransfer',
state__in=(OrderPayment.PAYMENT_STATE_CREATED, OrderPayment.PAYMENT_STATE_PENDING),
defaults={
'state': OrderPayment.PAYMENT_STATE_CREATED,
}
)[0]
p.info_data = {
'reference': trans.reference,
'date': trans.date,
'payer': trans.payer,
'iban': trans.iban,
'bic': trans.bic,
'trans_id': trans.pk
}
try:
p.confirm(user=self.request.user)
except Quota.QuotaExceededException:
pass
except SendMailException:
return JsonResponse({
'status': 'error',
'message': _('Problem sending email.')
})
trans.state = BankTransaction.STATE_VALID
trans.save()
trans.order.payments.filter(
provider='banktransfer',
state__in=(OrderPayment.PAYMENT_STATE_CREATED, OrderPayment.PAYMENT_STATE_PENDING),
).update(state=OrderPayment.PAYMENT_STATE_CANCELED)
return JsonResponse({
'status': 'ok',
})
def _assign(self, trans, code):
try:
if '-' in code:
trans.order = self.order_qs().get(code=code.rsplit('-', 1)[1], event__slug__iexact=code.rsplit('-', 1)[0])
else:
trans.order = self.order_qs().get(code=code.rsplit('-', 1)[-1])
if trans.currency and trans.order and trans.currency != trans.order.event.currency:
return JsonResponse({
'status': 'error',
'message': _('Currencies do not match.')
})
except Order.DoesNotExist:
return JsonResponse({
'status': 'error',
'message': _('Unknown order code')
})
else:
return self._retry(trans)
def _comment(self, trans, comment):
from pretix.base.templatetags.rich_text import rich_text
trans.comment = comment
trans.save()
return JsonResponse({
'status': 'ok',
'comment': rich_text(comment),
'plain': comment,
})
def post(self, request, *args, **kwargs):
for k, v in request.POST.items():
if not k.startswith('action_'):
continue
if 'event' in kwargs:
trans = get_object_or_404(BankTransaction, id=k.split('_')[1], event=request.event)
else:
trans = get_object_or_404(BankTransaction, id=k.split('_')[1], organizer=request.organizer)
if v == 'discard' and trans.state in (BankTransaction.STATE_INVALID, BankTransaction.STATE_ERROR,
BankTransaction.STATE_NOMATCH, BankTransaction.STATE_DUPLICATE):
return self._discard(trans)
elif v == 'accept' and trans.state == BankTransaction.STATE_INVALID:
# Accept anyway even with wrong amount
return self._accept_ignore_amount(trans)
elif v.startswith('comment:'):
return self._comment(trans, v[8:])
elif v.startswith('assign:') and trans.state in (BankTransaction.STATE_NOMATCH,
BankTransaction.STATE_DUPLICATE):
return self._assign(trans, v[7:])
elif v == 'retry' and trans.state in (BankTransaction.STATE_ERROR, BankTransaction.STATE_DUPLICATE):
return self._retry(trans)
return JsonResponse({
'status': 'error',
'message': 'Unknown action'
})
def get(self, request, *args, **kwargs):
u = request.GET.get('query', '')
if len(u) < 2:
return JsonResponse({'results': []})
if "-" in u:
code = (Q(event__slug__icontains=u.split("-")[0])
& Q(code__icontains=Order.normalize_code(u.split("-")[1])))
else:
code = Q(code__icontains=Order.normalize_code(u))
invoice_nos = {u, u.upper()}
if u.isdigit():
if hasattr(request, 'event'):
invoice_nos.add(u.zfill(request.event.settings.invoice_numbers_counter_length))
else:
for i in range(2, 12):
invoice_nos.add(u.zfill(i))
qs = self.order_qs().order_by('pk').filter(
code
| Q(email__icontains=u)
| Q(all_positions__attendee_name_cached__icontains=u)
| Q(all_positions__attendee_email__icontains=u)
| Q(invoice_address__name_cached__icontains=u)
| Q(invoice_address__company__icontains=u)
| Q(invoices__invoice_no__in=invoice_nos)
| Q(invoices__full_invoice_no=u)
).select_related('event').annotate(pcnt=Count('invoices')).distinct()
# Yep, we wouldn't need to count the invoices here. However, having this Count() statement in there
# tricks Django into generating a GROUP BY clause that it otherwise wouldn't and that is required to
# avoid duplicate results. Yay?
return JsonResponse({
'results': [
{
'code': o.event.slug.upper() + '-' + o.code,
'status': o.get_status_display(),
'total': money_filter(o.total, o.event.currency)
} for o in qs
]
})
def order_qs(self):
return self.request.event.orders
class JobDetailView(DetailView):
template_name = 'pretixplugins/banktransfer/job_detail.html'
permission = 'can_change_orders'
context_objectname = 'job'
def redirect_form(self):
kwargs = {
'organizer': self.request.organizer.slug,
}
if 'event' in self.kwargs:
kwargs['event'] = self.kwargs['event']
return redirect(reverse('plugins:banktransfer:import', kwargs=kwargs))
def redirect_back(self):
kwargs = {
'organizer': self.request.organizer.slug,
'job': self.kwargs['job']
}
if 'event' in self.kwargs:
kwargs['event'] = self.kwargs['event']
return redirect(reverse('plugins:banktransfer:import.job', kwargs=kwargs))
@cached_property
def job(self):
if 'event' in self.kwargs:
kwargs = {'event': self.request.event}
else:
kwargs = {'organizer': self.request.organizer}
return get_object_or_404(BankImportJob, id=self.kwargs['job'], **kwargs)
def get(self, request, *args, **kwargs):
if 'ajax' in request.GET:
return JsonResponse({
'state': self.job.state
})
context = self.get_context_data()
return self.render_to_response(context)
def get_context_data(self, **kwargs):
ctx = {}
qs = self.job.transactions.select_related('order', 'order__event')
ctx['transactions_valid'] = qs.filter(state=BankTransaction.STATE_VALID).count()
ctx['transactions_invalid'] = qs.filter(state__in=[
BankTransaction.STATE_INVALID, BankTransaction.STATE_ERROR
]).count()
ctx['transactions_ignored'] = qs.filter(state__in=[
BankTransaction.STATE_DUPLICATE, BankTransaction.STATE_NOMATCH
]).count()
ctx['job'] = self.job
ctx['organizer'] = self.request.organizer
if 'event' in self.kwargs:
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base.html'
else:
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base_organizer.html'
return ctx
class BankTransactionFilterForm(forms.Form):
search_text = forms.CharField(required=False, widget=forms.TextInput(attrs={'class': "form-control", "placeholder": _("Search text")}))
amount_min = forms.DecimalField(required=False, localize=True, widget=forms.TextInput(attrs={'class': "form-control", "placeholder": _("min"), "size": 8}))
amount_max = forms.DecimalField(required=False, localize=True, widget=forms.TextInput(attrs={'class': "form-control", "placeholder": _("max"), "size": 8}))
date_min = forms.DateField(required=False, widget=DatePickerWidget(attrs={"size": 8}))
date_max = forms.DateField(required=False, widget=DatePickerWidget(attrs={"size": 8}))
def is_valid(self):
return super().is_valid() and any(value for value in self.cleaned_data.values())
def filter(self, qs):
if not self.is_valid():
raise ValueError(_("Filter form is not valid."))
if self.cleaned_data.get('search_text'):
q = self.cleaned_data['search_text']
qs = qs.filter(Q(payer__icontains=q) | Q(reference__icontains=q) | Q(comment__icontains=q) | Q(iban__icontains=q) | Q(bic__icontains=q))
if self.cleaned_data.get('amount_min') is not None:
qs = qs.filter(amount__gte=self.cleaned_data['amount_min'])
if self.cleaned_data.get("amount_max") is not None:
qs = qs.filter(amount__lte=self.cleaned_data['amount_max'])
if self.cleaned_data.get('date_min') is not None:
qs = qs.filter(date_parsed__gte=self.cleaned_data['date_min'])
if self.cleaned_data.get('date_max') is not None:
qs = qs.filter(date_parsed__lte=self.cleaned_data['date_max'])
return qs
class ImportView(ListView):
template_name = 'pretixplugins/banktransfer/import_form.html'
permission = 'can_change_orders'
context_object_name = 'transactions_unhandled'
paginate_by = 30
def get_queryset(self):
if 'event' in self.kwargs:
qs = BankTransaction.objects.filter(
Q(event=self.request.event)
)
else:
qs = BankTransaction.objects.filter(
Q(organizer=self.request.organizer)
)
qs = qs.select_related('order').filter(state__in=[
BankTransaction.STATE_INVALID, BankTransaction.STATE_ERROR,
BankTransaction.STATE_DUPLICATE, BankTransaction.STATE_NOMATCH
])
filter_form = BankTransactionFilterForm(self.request.GET or None)
if filter_form.is_valid():
qs = filter_form.filter(qs)
return qs.order_by('-import_job__created')
def discard_all(self):
self.get_queryset().update(payer='', reference='', state=BankTransaction.STATE_DISCARDED)
messages.success(self.request, _('All unresolved transactions have been discarded.'))
def post(self, *args, **kwargs):
if self.request.POST.get('discard', '') == 'all':
self.discard_all()
return self.redirect_back()
elif ('file' in self.request.FILES and '.csv' in self.request.FILES.get('file').name.lower()) or 'amount' in self.request.POST:
# Process CSV
return self.process_csv()
elif 'file' in self.request.FILES and (
'.txt' in self.request.FILES.get('file').name.lower()
or '.sta' in self.request.FILES.get('file').name.lower()
or '.mta' in self.request.FILES.get('file').name.lower() # Volksbank's structure
or '.swi' in self.request.FILES.get('file').name.lower() # Rabobank's MT940 Structured
):
return self.process_mt940()
elif self.request.FILES.get('file') is None:
messages.error(self.request, _('You must choose a file to import.'))
return self.redirect_back()
else:
messages.error(self.request, _('We were unable to detect the file type of this import. Please '
'contact support for help.'))
return self.redirect_back()
@cached_property
def settings(self):
return SettingsSandbox('payment', 'banktransfer', getattr(self.request, 'event', self.request.organizer))
def process_mt940(self):
try:
return self.start_processing(mt940import.parse(self.request.FILES.get('file')))
except:
logger.exception('Failed to import MT940 file')
messages.error(self.request, _('We were unable to process your input.'))
return self.redirect_back()
def _hint_settings_name(self, currency):
if len(self.currencies) > 1:
return f'banktransfer_csvhint_{currency}'
return 'banktransfer_csvhint'
def process_csv_file(self):
o = getattr(self.request, 'event', self.request.organizer)
try:
data = csvimport.get_rows_from_file(self.request.FILES['file'])
except csv.Error as e: # TODO: narrow down
logger.error('Import failed: ' + str(e))
messages.error(self.request, _('I\'m sorry, but we were unable to import this CSV file. Please '
'contact support for help.'))
return self.redirect_back()
if len(data) == 0:
messages.error(self.request, _('I\'m sorry, but we detected this file as empty. Please '
'contact support for help.'))
if o.settings.get(self._hint_settings_name(self.request.POST.get('currency'))) is not None:
hint = o.settings.get(self._hint_settings_name(self.request.POST.get('currency')), as_type=dict)
try:
parsed, good = csvimport.parse(data, hint)
except csvimport.HintMismatchError: # TODO: narrow down
logger.exception('Import using stored hint failed')
else:
if good:
return self.start_processing(parsed)
return self.assign_view(data)
def process_csv_hint(self):
try:
data = json.loads(self.request.POST.get('data').strip())
except ValueError:
messages.error(self.request, _('Invalid input data.'))
return self.get(self.request, *self.args, **self.kwargs)
if 'reference' not in self.request.POST:
messages.error(self.request, _('You need to select the column containing the payment reference.'))
return self.assign_view(data)
try:
hint = csvimport.new_hint(self.request.POST)
except Exception as e:
logger.error('Parsing hint failed: ' + str(e))
messages.error(self.request, _('We were unable to process your input.'))
return self.assign_view(data)
o = getattr(self.request, 'event', self.request.organizer)
try:
o.settings.set(self._hint_settings_name(self.request.POST.get('currency')), hint)
except Exception as e: # TODO: narrow down
logger.error('Import using stored hint failed: ' + str(e))
pass
else:
parsed, __ = csvimport.parse(data, hint)
return self.start_processing(parsed)
def process_csv(self):
if 'file' in self.request.FILES:
return self.process_csv_file()
elif 'amount' in self.request.POST:
return self.process_csv_hint()
return super().get(self.request)
def assign_view(self, parsed):
ctx = {
'json': json.dumps(parsed),
'rows': parsed,
}
if 'event' in self.kwargs:
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base.html'
else:
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base_organizer.html'
ctx['organizer'] = self.request.organizer
return render(self.request, 'pretixplugins/banktransfer/import_assign.html', ctx)
@cached_property
def job_running(self):
if 'event' in self.kwargs:
qs = BankImportJob.objects.filter(
Q(event=self.request.event) | Q(organizer=self.request.organizer)
)
else:
qs = BankImportJob.objects.filter(
Q(organizer=self.request.organizer)
)
return qs.filter(
state=BankImportJob.STATE_RUNNING,
created__lte=now() - timedelta(minutes=30) # safety timeout
).first()
def redirect_back(self):
kwargs = {
'organizer': self.request.organizer.slug
}
if 'event' in self.kwargs:
kwargs['event'] = self.kwargs['event']
return redirect(reverse('plugins:banktransfer:import', kwargs=kwargs))
@cached_property
def currencies(self):
if hasattr(self.request, 'event'):
return [self.request.event.currency]
else:
return list(
self.request.organizer.events.order_by('currency').values_list('currency', flat=True).distinct()
)
def start_processing(self, parsed):
if self.job_running:
messages.error(self.request,
_('An import is currently being processed, please try again in a few minutes.'))
return self.redirect_back()
if 'event' in self.kwargs:
job = BankImportJob.objects.create(event=self.request.event, organizer=self.request.organizer)
else:
if len(self.currencies) != 1:
currency = self.request.POST.get("currency")
if not currency or currency not in self.currencies:
messages.error(self.request,
_('No currency has been selected.'))
return self.redirect_back()
job = BankImportJob.objects.create(organizer=self.request.organizer, currency=currency)
else:
job = BankImportJob.objects.create(organizer=self.request.organizer, currency=self.currencies[0])
process_banktransfers.apply_async(kwargs={
'job': job.pk,
'data': parsed
})
kwargs = {
'organizer': self.request.organizer.slug,
'job': job.pk
}
if 'event' in self.kwargs:
kwargs['event'] = self.kwargs['event']
return redirect(reverse('plugins:banktransfer:import.job', kwargs=kwargs))
def get_context_data(self, **kwargs):
ctx = super().get_context_data()
ctx['job_running'] = self.job_running
ctx['no_more_payments'] = False
ctx['filter_form'] = BankTransactionFilterForm(self.request.GET or None)
if not hasattr(self.request, 'event'):
ctx['currencies'] = self.currencies
if 'event' in self.kwargs:
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base.html'
if not self.request.event.has_subevents and self.request.event.settings.get('payment_term_last'):
if now() > self.request.event.payment_term_last:
ctx['no_more_payments'] = True
ctx['lastimport'] = BankImportJob.objects.filter(
state=BankImportJob.STATE_COMPLETED,
organizer=self.request.organizer,
event=self.request.event
).order_by('created').last()
ctx['runningimport'] = BankImportJob.objects.filter(
state__in=[BankImportJob.STATE_PENDING, BankImportJob.STATE_RUNNING],
organizer=self.request.organizer,
event=self.request.event
).order_by('created').last()
else:
ctx['lastimport'] = BankImportJob.objects.filter(
state=BankImportJob.STATE_COMPLETED,
organizer=self.request.organizer,
event__isnull=True
).order_by('created').last()
ctx['runningimport'] = BankImportJob.objects.filter(
state__in=[BankImportJob.STATE_PENDING, BankImportJob.STATE_RUNNING],
organizer=self.request.organizer,
event__isnull=True
).order_by('created').last()
ctx['basetpl'] = 'pretixplugins/banktransfer/import_base_organizer.html'
ctx['organizer'] = self.request.organizer
return ctx
class OrganizerBanktransferView:
def dispatch(self, request, *args, **kwargs):
return super().dispatch(request, *args, **kwargs)
class EventImportView(EventPermissionRequiredMixin, ImportView):
permission = 'can_change_orders'
class OrganizerImportView(OrganizerBanktransferView, OrganizerPermissionRequiredMixin, OrganizerDetailViewMixin,
ImportView):
permission = 'can_change_orders'
class EventJobDetailView(EventPermissionRequiredMixin, JobDetailView):
permission = 'can_change_orders'
class OrganizerJobDetailView(OrganizerBanktransferView, OrganizerPermissionRequiredMixin, OrganizerDetailViewMixin,
JobDetailView):
permission = 'can_change_orders'
class EventActionView(EventPermissionRequiredMixin, ActionView):
permission = 'can_change_orders'
class OrganizerActionView(OrganizerBanktransferView, OrganizerPermissionRequiredMixin, OrganizerDetailViewMixin,
ActionView):
permission = 'can_change_orders'
def order_qs(self):
all = self.request.user.teams.filter(organizer=self.request.organizer, can_change_orders=True,
can_view_orders=True, all_events=True).exists()
if self.request.user.has_active_staff_session(self.request.session.session_key) or all:
return Order.objects.filter(event__organizer=self.request.organizer)
else:
return Order.objects.filter(
event_id__in=self.request.user.teams.filter(
organizer=self.request.organizer, can_change_orders=True, can_view_orders=True
).values_list('limit_events__id', flat=True)
)
def _row_key_func(row):
return row['iban'], row['bic']
def _unite_transaction_rows(transaction_rows):
united_transactions_rows = []
transaction_rows = sorted(transaction_rows, key=_row_key_func)
for (iban, bic), group in itertools.groupby(transaction_rows, _row_key_func):
rows = list(group)
united_transactions_rows.append({
"iban": iban,
"bic": bic,
"id": ", ".join(sorted(set(r['id'] for r in rows))),
"payer": ", ".join(sorted(set(r['payer'] for r in rows))),
"amount": sum(r['amount'] for r in rows),
"comment": ", ".join(r['comment'] for r in rows if r.get('comment')) or None,
})
return united_transactions_rows
class RefundExportListView(ListView):
template_name = 'pretixplugins/banktransfer/refund_export.html'
model = RefundExport
context_object_name = 'exports'
def get_success_url(self):
raise NotImplementedError
def get_unexported(self) -> QuerySet:
raise NotImplementedError()
def get_context_data(self, **kwargs):
ctx = super().get_context_data()
ctx['num_new'] = self.get_unexported().count()
ctx['basetpl'] = "pretixcontrol/event/base.html"
if not hasattr(self.request, 'event'):
ctx['basetpl'] = "pretixcontrol/organizers/base.html"
return ctx
@transaction.atomic()
def post(self, request, *args, **kwargs):
unite_transactions = request.POST.get("unite_transactions", False)
valid_refunds: Set[OrderRefund] = set()
for refund in self.get_unexported().select_related('order', 'order__event'):
if not refund.info_data:
# Should not happen
messages.warning(request,
_("We could not find bank account information for the refund {refund_id}. It was marked as failed.")
.format(refund_id=refund.full_id))
refund.state = OrderRefund.REFUND_STATE_FAILED
refund.save()
continue
else:
valid_refunds.add(refund)
if valid_refunds:
transaction_rows = defaultdict(list)
for refund in valid_refunds:
data = refund.info_data
transaction_rows[refund.order.event.currency].append({
"amount": refund.amount,
"id": refund.full_id,
"comment": refund.comment,
**{key: data.get(key) for key in ("payer", "iban", "bic")}
})
refund.done(user=self.request.user)
if unite_transactions:
for currency, rows in transaction_rows.items():
transaction_rows[currency] = _unite_transaction_rows(rows)
for currency, rows in transaction_rows.items():
rows_data = json.dumps(rows, cls=CustomJSONEncoder)
if hasattr(request, 'event'):
RefundExport.objects.create(event=self.request.event, testmode=self.request.event.testmode, rows=rows_data, currency=currency)
else:
RefundExport.objects.create(organizer=self.request.organizer, testmode=False, rows=rows_data, currency=currency)
else:
messages.warning(request, _('No valid orders have been found.'))
return redirect(self.get_success_url())
class EventRefundExportListView(EventPermissionRequiredMixin, RefundExportListView):
permission = 'can_change_orders'
def get_success_url(self):
return reverse('plugins:banktransfer:refunds.list', kwargs={
'event': self.request.event.slug,
'organizer': self.request.organizer.slug,
})
def get_queryset(self):
return RefundExport.objects.filter(
event=self.request.event
).order_by('-datetime')
def get_unexported(self):
return OrderRefund.objects.filter(
order__event=self.request.event,
provider__in=['banktransfer', 'sepadebit'],
state=OrderRefund.REFUND_STATE_CREATED,
order__testmode=self.request.event.testmode,
)
class OrganizerRefundExportListView(OrganizerPermissionRequiredMixin, RefundExportListView):
permission = 'can_change_orders'
def get_success_url(self):
return reverse('plugins:banktransfer:refunds.list', kwargs={
'organizer': self.request.organizer.slug,
})
def get_queryset(self):
return RefundExport.objects.filter(
Q(organizer=self.request.organizer) | Q(event__organizer=self.request.organizer)
).order_by('-datetime')
def get_unexported(self):
return OrderRefund.objects.filter(
order__event__organizer=self.request.organizer,
provider__in=['banktransfer', 'sepadebit'],
state=OrderRefund.REFUND_STATE_CREATED,
order__testmode=False,
)
class DownloadRefundExportView(DetailView):
model = RefundExport
def get(self, request, *args, **kwargs):
self.object: RefundExport = self.get_object()
self.object.downloaded = True
self.object.save(update_fields=["downloaded"])
filename, content_type, data = get_refund_export_csv(self.object)
return FileResponse(data, as_attachment=True, filename=filename, content_type=content_type)
class EventDownloadRefundExportView(EventPermissionRequiredMixin, DownloadRefundExportView):
permission = 'can_change_orders'
def get_object(self, *args, **kwargs):
return get_object_or_404(
RefundExport,
event=self.request.event,
pk=self.kwargs.get('id')
)
class OrganizerDownloadRefundExportView(OrganizerPermissionRequiredMixin, OrganizerDetailViewMixin, DownloadRefundExportView):
permission = 'can_change_orders'
def get_object(self, *args, **kwargs):
return get_object_or_404(
RefundExport,
organizer=self.request.organizer,
pk=self.kwargs.get('id')
)
class SepaXMLExportForm(forms.Form):
account_holder = forms.CharField(label=_("Account holder"))
iban = IBANFormField(label="IBAN")
bic = BICFormField(label="BIC")
def set_initial_from_event(self, event: Event):
banktransfer = BankTransfer(event)
self.initial["account_holder"] = banktransfer.settings.get("bank_details_sepa_name")
self.initial["iban"] = banktransfer.settings.get("bank_details_sepa_iban")
self.initial["bic"] = banktransfer.settings.get("bank_details_sepa_bic")
class SepaXMLExportView(SingleObjectMixin, FormView):
form_class = SepaXMLExportForm
model = RefundExport
template_name = 'pretixplugins/banktransfer/sepa_export.html'
context_object_name = "export"
def setup(self, request, *args, **kwargs):
super().setup(request, *args, **kwargs)
self.object: RefundExport = self.get_object()
def form_valid(self, form):
self.object.downloaded = True
self.object.save(update_fields=["downloaded"])
filename, content_type, data = build_sepa_xml(self.object, **form.cleaned_data)
return FileResponse(data, as_attachment=True, filename=filename, content_type=content_type)
def get_context_data(self, **kwargs):
ctx = super().get_context_data()
ctx['basetpl'] = "pretixcontrol/event/base.html"
if not hasattr(self.request, 'event'):
ctx['basetpl'] = "pretixcontrol/organizers/base.html"
return ctx
class EventSepaXMLExportView(EventPermissionRequiredMixin, SepaXMLExportView):
permission = 'can_change_orders'
def get_object(self, *args, **kwargs):
return get_object_or_404(
RefundExport,
event=self.request.event,
pk=self.kwargs.get('id')
)
def get_form(self, form_class=None):
form = super().get_form(form_class)
form.set_initial_from_event(self.object.event)
return form
class OrganizerSepaXMLExportView(OrganizerPermissionRequiredMixin, OrganizerDetailViewMixin, SepaXMLExportView):
permission = 'can_change_orders'
def get_object(self, *args, **kwargs):
return get_object_or_404(
RefundExport,
organizer=self.request.organizer,
pk=self.kwargs.get('id')
)
@method_decorator(xframe_options_exempt, 'dispatch')
class SendInvoiceMailView(EventViewMixin, OrderDetailMixin, View):
def post(self, request, *args, **kwargs):
if not self.order:
raise Http404(_('Unknown order code or not authorized to access this order.'))
try:
validate_email(request.POST['email'])
except ValidationError:
messages.error(request, _('Please enter a valid email address.'))
return redirect(self.get_order_url())
last_payment = self.order.payments.last()
if (not last_payment
or last_payment.provider != BankTransfer.identifier
or last_payment.state != OrderPayment.PAYMENT_STATE_CREATED):
messages.error(request, _('No pending bank transfer payment found. Maybe the order has been paid already?'))
return redirect(self.get_order_url())
if not last_payment.payment_provider.settings.get('invoice_email', as_type=bool):
messages.error(request, _('Sending invoices via email is disabled by the event organizer.'))
return redirect(self.get_order_url())
last_invoice = self.order.invoices.last()
if not last_invoice:
messages.error(request, _('No invoice found, please request an invoice first.'))
return redirect(self.get_order_url())
provider = last_payment.payment_provider
provider.send_invoice_to_alternate_email(self.order, last_invoice, request.POST['email'])
last_payment.info_data = {
**last_payment.info_data,
'send_invoice_to': request.POST['email'],
}
last_payment.save(update_fields=['info'])
messages.success(request, _('Sending the latest invoice via e-mail to {email}.').format(email=request.POST['email']))
return redirect(self.get_order_url())