mirror of
https://github.com/pretix/pretix.git
synced 2026-04-26 23:52:35 +00:00
Compare commits
25 Commits
timemachin
...
plugin-ava
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9ab5a17ac9 | ||
|
|
b24664c73e | ||
|
|
0dbbc38ca0 | ||
|
|
f3597f1a44 | ||
|
|
2e01887e79 | ||
|
|
5a7e7fbde3 | ||
|
|
7b296107c5 | ||
|
|
4f449ce6b4 | ||
|
|
e6ea8fb5bf | ||
|
|
547910beec | ||
|
|
eef1560ede | ||
|
|
3d68bbb619 | ||
|
|
dc4556d428 | ||
|
|
5099fa16e0 | ||
|
|
f3fb1e66dc | ||
|
|
99e9690d48 | ||
|
|
e63e82e854 | ||
|
|
c662e627d5 | ||
|
|
f2121c7853 | ||
|
|
3ce6dbf798 | ||
|
|
43b91af5e6 | ||
|
|
034d6b997e | ||
|
|
345ad35fcf | ||
|
|
347337e76f | ||
|
|
c07ba31307 |
@@ -1719,6 +1719,56 @@ List of all order positions
|
|||||||
:statuscode 401: Authentication failure
|
:statuscode 401: Authentication failure
|
||||||
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to view this resource.
|
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to view this resource.
|
||||||
|
|
||||||
|
.. http:get:: /api/v1/organizers/(organizer)/orderpositions/
|
||||||
|
|
||||||
|
Returns a list of all order positions within all events of a given organizer (with sufficient access permissions).
|
||||||
|
|
||||||
|
The supported query parameters and output format of this endpoint are almost identical to those of the list endpoint
|
||||||
|
within an event.
|
||||||
|
The only changes are that responses also contain the ``event`` attribute in each result and that the 'pdf_data'
|
||||||
|
parameter is not supported.
|
||||||
|
|
||||||
|
**Example request**:
|
||||||
|
|
||||||
|
.. sourcecode:: http
|
||||||
|
|
||||||
|
GET /api/v1/organizers/bigevents/orderpositions/ HTTP/1.1
|
||||||
|
Host: pretix.eu
|
||||||
|
Accept: application/json, text/javascript
|
||||||
|
|
||||||
|
**Example response**:
|
||||||
|
|
||||||
|
.. sourcecode:: http
|
||||||
|
|
||||||
|
HTTP/1.1 200 OK
|
||||||
|
Vary: Accept
|
||||||
|
Content-Type: application/json
|
||||||
|
X-Page-Generated: 2017-12-01T10:00:00Z
|
||||||
|
|
||||||
|
{
|
||||||
|
"count": 1,
|
||||||
|
"next": null,
|
||||||
|
"previous": null,
|
||||||
|
"results": [
|
||||||
|
{
|
||||||
|
"id:": 23442
|
||||||
|
"event": "sampleconf",
|
||||||
|
"order": "ABC12",
|
||||||
|
"positionid": 1,
|
||||||
|
"canceled": false,
|
||||||
|
"item": 1345,
|
||||||
|
...
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
:param organizer: The ``slug`` field of the organizer to fetch
|
||||||
|
:statuscode 200: no error
|
||||||
|
:statuscode 401: Authentication failure
|
||||||
|
:statuscode 403: The requested organizer/event does not exist **or** you have no permission to view this resource.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Fetching individual positions
|
Fetching individual positions
|
||||||
-----------------------------
|
-----------------------------
|
||||||
|
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ dependencies = [
|
|||||||
"phonenumberslite==9.0.*",
|
"phonenumberslite==9.0.*",
|
||||||
"Pillow==12.1.*",
|
"Pillow==12.1.*",
|
||||||
"pretix-plugin-build",
|
"pretix-plugin-build",
|
||||||
"protobuf==6.33.*",
|
"protobuf==7.34.*",
|
||||||
"psycopg2-binary",
|
"psycopg2-binary",
|
||||||
"pycountry",
|
"pycountry",
|
||||||
"pycparser==3.0",
|
"pycparser==3.0",
|
||||||
@@ -92,7 +92,7 @@ dependencies = [
|
|||||||
"redis==7.1.*",
|
"redis==7.1.*",
|
||||||
"reportlab==4.4.*",
|
"reportlab==4.4.*",
|
||||||
"requests==2.32.*",
|
"requests==2.32.*",
|
||||||
"sentry-sdk==2.53.*",
|
"sentry-sdk==2.54.*",
|
||||||
"sepaxml==2.7.*",
|
"sepaxml==2.7.*",
|
||||||
"stripe==7.9.*",
|
"stripe==7.9.*",
|
||||||
"text-unidecode==1.*",
|
"text-unidecode==1.*",
|
||||||
|
|||||||
@@ -115,10 +115,10 @@ class PluginsField(serializers.Field):
|
|||||||
|
|
||||||
def to_representation(self, obj):
|
def to_representation(self, obj):
|
||||||
from pretix.base.plugins import get_all_plugins
|
from pretix.base.plugins import get_all_plugins
|
||||||
|
active_plugins = set(obj.get_plugins())
|
||||||
return sorted([
|
return sorted([
|
||||||
p.module for p in get_all_plugins()
|
p.module for p in get_all_plugins()
|
||||||
if not p.name.startswith('.') and getattr(p, 'visible', True) and p.module in obj.get_plugins()
|
if not p.name.startswith('.') and getattr(p, 'visible', True) and p.module in active_plugins
|
||||||
])
|
])
|
||||||
|
|
||||||
def to_internal_value(self, data):
|
def to_internal_value(self, data):
|
||||||
|
|||||||
@@ -637,6 +637,14 @@ class OrderPositionSerializer(I18nAwareModelSerializer):
|
|||||||
return entry
|
return entry
|
||||||
|
|
||||||
|
|
||||||
|
class OrganizerOrderPositionSerializer(OrderPositionSerializer):
|
||||||
|
event = SlugRelatedField(slug_field='slug', read_only=True)
|
||||||
|
|
||||||
|
class Meta(OrderPositionSerializer.Meta):
|
||||||
|
fields = OrderPositionSerializer.Meta.fields + ('event',)
|
||||||
|
read_only_fields = OrderPositionSerializer.Meta.read_only_fields + ('event',)
|
||||||
|
|
||||||
|
|
||||||
class RequireAttentionField(serializers.Field):
|
class RequireAttentionField(serializers.Field):
|
||||||
def to_representation(self, instance: OrderPosition):
|
def to_representation(self, instance: OrderPosition):
|
||||||
return instance.require_checkin_attention
|
return instance.require_checkin_attention
|
||||||
|
|||||||
@@ -67,6 +67,7 @@ orga_router.register(r'invoices', order.InvoiceViewSet)
|
|||||||
orga_router.register(r'scheduled_exports', exporters.ScheduledOrganizerExportViewSet)
|
orga_router.register(r'scheduled_exports', exporters.ScheduledOrganizerExportViewSet)
|
||||||
orga_router.register(r'exporters', exporters.OrganizerExportersViewSet, basename='exporters')
|
orga_router.register(r'exporters', exporters.OrganizerExportersViewSet, basename='exporters')
|
||||||
orga_router.register(r'transactions', order.OrganizerTransactionViewSet)
|
orga_router.register(r'transactions', order.OrganizerTransactionViewSet)
|
||||||
|
orga_router.register(r'orderpositions', order.OrganizerOrderPositionViewSet, basename='orderpositions')
|
||||||
|
|
||||||
team_router = routers.DefaultRouter()
|
team_router = routers.DefaultRouter()
|
||||||
team_router.register(r'members', organizer.TeamMemberViewSet)
|
team_router.register(r'members', organizer.TeamMemberViewSet)
|
||||||
@@ -83,7 +84,7 @@ event_router.register(r'discounts', discount.DiscountViewSet)
|
|||||||
event_router.register(r'quotas', item.QuotaViewSet)
|
event_router.register(r'quotas', item.QuotaViewSet)
|
||||||
event_router.register(r'vouchers', voucher.VoucherViewSet)
|
event_router.register(r'vouchers', voucher.VoucherViewSet)
|
||||||
event_router.register(r'orders', order.EventOrderViewSet)
|
event_router.register(r'orders', order.EventOrderViewSet)
|
||||||
event_router.register(r'orderpositions', order.OrderPositionViewSet)
|
event_router.register(r'orderpositions', order.EventOrderPositionViewSet)
|
||||||
event_router.register(r'transactions', order.TransactionViewSet)
|
event_router.register(r'transactions', order.TransactionViewSet)
|
||||||
event_router.register(r'invoices', order.InvoiceViewSet)
|
event_router.register(r'invoices', order.InvoiceViewSet)
|
||||||
event_router.register(r'revokedsecrets', order.RevokedSecretViewSet, basename='revokedsecrets')
|
event_router.register(r'revokedsecrets', order.RevokedSecretViewSet, basename='revokedsecrets')
|
||||||
|
|||||||
@@ -57,9 +57,10 @@ from pretix.api.serializers.order import (
|
|||||||
BlockedTicketSecretSerializer, InvoiceSerializer, OrderCreateSerializer,
|
BlockedTicketSecretSerializer, InvoiceSerializer, OrderCreateSerializer,
|
||||||
OrderPaymentCreateSerializer, OrderPaymentSerializer,
|
OrderPaymentCreateSerializer, OrderPaymentSerializer,
|
||||||
OrderPositionSerializer, OrderRefundCreateSerializer,
|
OrderPositionSerializer, OrderRefundCreateSerializer,
|
||||||
OrderRefundSerializer, OrderSerializer, OrganizerTransactionSerializer,
|
OrderRefundSerializer, OrderSerializer, OrganizerOrderPositionSerializer,
|
||||||
PriceCalcSerializer, PrintLogSerializer, RevokedTicketSecretSerializer,
|
OrganizerTransactionSerializer, PriceCalcSerializer, PrintLogSerializer,
|
||||||
SimulatedOrderSerializer, TransactionSerializer,
|
RevokedTicketSecretSerializer, SimulatedOrderSerializer,
|
||||||
|
TransactionSerializer,
|
||||||
)
|
)
|
||||||
from pretix.api.serializers.orderchange import (
|
from pretix.api.serializers.orderchange import (
|
||||||
BlockNameSerializer, OrderChangeOperationSerializer,
|
BlockNameSerializer, OrderChangeOperationSerializer,
|
||||||
@@ -1065,8 +1066,7 @@ with scopes_disabled():
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class OrderPositionViewSet(viewsets.ModelViewSet):
|
class OrderPositionViewSetMixin:
|
||||||
serializer_class = OrderPositionSerializer
|
|
||||||
queryset = OrderPosition.all.none()
|
queryset = OrderPosition.all.none()
|
||||||
filter_backends = (DjangoFilterBackend, RichOrderingFilter)
|
filter_backends = (DjangoFilterBackend, RichOrderingFilter)
|
||||||
ordering = ('order__datetime', 'positionid')
|
ordering = ('order__datetime', 'positionid')
|
||||||
@@ -1087,8 +1087,7 @@ class OrderPositionViewSet(viewsets.ModelViewSet):
|
|||||||
|
|
||||||
def get_serializer_context(self):
|
def get_serializer_context(self):
|
||||||
ctx = super().get_serializer_context()
|
ctx = super().get_serializer_context()
|
||||||
ctx['event'] = self.request.event
|
ctx['pdf_data'] = False
|
||||||
ctx['pdf_data'] = self.request.query_params.get('pdf_data', 'false').lower() == 'true'
|
|
||||||
ctx['check_quotas'] = self.request.query_params.get('check_quotas', 'true').lower() == 'true'
|
ctx['check_quotas'] = self.request.query_params.get('check_quotas', 'true').lower() == 'true'
|
||||||
return ctx
|
return ctx
|
||||||
|
|
||||||
@@ -1097,9 +1096,8 @@ class OrderPositionViewSet(viewsets.ModelViewSet):
|
|||||||
qs = OrderPosition.all
|
qs = OrderPosition.all
|
||||||
else:
|
else:
|
||||||
qs = OrderPosition.objects
|
qs = OrderPosition.objects
|
||||||
|
qs = qs.filter(order__event__organizer=self.request.organizer)
|
||||||
qs = qs.filter(order__event=self.request.event)
|
if self.request.query_params.get('pdf_data', 'false').lower() == 'true' and getattr(self.request, 'event', None):
|
||||||
if self.request.query_params.get('pdf_data', 'false').lower() == 'true':
|
|
||||||
prefetch_related_objects([self.request.organizer], 'meta_properties')
|
prefetch_related_objects([self.request.organizer], 'meta_properties')
|
||||||
prefetch_related_objects(
|
prefetch_related_objects(
|
||||||
[self.request.event],
|
[self.request.event],
|
||||||
@@ -1154,9 +1152,9 @@ class OrderPositionViewSet(viewsets.ModelViewSet):
|
|||||||
qs = qs.prefetch_related(
|
qs = qs.prefetch_related(
|
||||||
Prefetch('checkins', queryset=Checkin.objects.select_related("device")),
|
Prefetch('checkins', queryset=Checkin.objects.select_related("device")),
|
||||||
Prefetch('print_logs', queryset=PrintLog.objects.select_related('device')),
|
Prefetch('print_logs', queryset=PrintLog.objects.select_related('device')),
|
||||||
'answers', 'answers__options', 'answers__question',
|
'answers', 'answers__options', 'answers__question', 'order__event', 'order__event__organizer'
|
||||||
).select_related(
|
).select_related(
|
||||||
'item', 'order', 'order__event', 'order__event__organizer', 'seat'
|
'item', 'order', 'seat'
|
||||||
)
|
)
|
||||||
return qs
|
return qs
|
||||||
|
|
||||||
@@ -1168,6 +1166,45 @@ class OrderPositionViewSet(viewsets.ModelViewSet):
|
|||||||
return prov
|
return prov
|
||||||
raise NotFound('Unknown output provider.')
|
raise NotFound('Unknown output provider.')
|
||||||
|
|
||||||
|
|
||||||
|
class OrganizerOrderPositionViewSet(OrderPositionViewSetMixin, viewsets.ReadOnlyModelViewSet):
|
||||||
|
serializer_class = OrganizerOrderPositionSerializer
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
qs = super().get_queryset()
|
||||||
|
|
||||||
|
perm = self.permission if self.request.method in SAFE_METHODS else self.write_permission
|
||||||
|
|
||||||
|
if isinstance(self.request.auth, (TeamAPIToken, Device)):
|
||||||
|
auth_obj = self.request.auth
|
||||||
|
elif self.request.user.is_authenticated:
|
||||||
|
auth_obj = self.request.user
|
||||||
|
else:
|
||||||
|
raise PermissionDenied("Unknown authentication scheme")
|
||||||
|
|
||||||
|
qs = qs.filter(
|
||||||
|
order__event__in=auth_obj.get_events_with_permission(perm, request=self.request).filter(
|
||||||
|
organizer=self.request.organizer
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return qs
|
||||||
|
|
||||||
|
|
||||||
|
class EventOrderPositionViewSet(OrderPositionViewSetMixin, viewsets.ModelViewSet):
|
||||||
|
serializer_class = OrderPositionSerializer
|
||||||
|
|
||||||
|
def get_serializer_context(self):
|
||||||
|
ctx = super().get_serializer_context()
|
||||||
|
ctx['event'] = self.request.event
|
||||||
|
ctx['pdf_data'] = self.request.query_params.get('pdf_data', 'false').lower() == 'true'
|
||||||
|
return ctx
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
qs = super().get_queryset()
|
||||||
|
qs = qs.filter(order__event=self.request.event)
|
||||||
|
return qs
|
||||||
|
|
||||||
@action(detail=True, methods=['POST'], url_name='price_calc')
|
@action(detail=True, methods=['POST'], url_name='price_calc')
|
||||||
def price_calc(self, request, *args, **kwargs):
|
def price_calc(self, request, *args, **kwargs):
|
||||||
"""
|
"""
|
||||||
|
|||||||
@@ -315,8 +315,9 @@ class OrderListExporter(MultiSheetListExporter):
|
|||||||
for id, vn in payment_methods:
|
for id, vn in payment_methods:
|
||||||
headers.append(_('Paid by {method}').format(method=vn))
|
headers.append(_('Paid by {method}').format(method=vn))
|
||||||
|
|
||||||
# get meta_data labels from first cached event
|
if self.event_object_cache:
|
||||||
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
|
# get meta_data labels from first cached event if any
|
||||||
|
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
|
||||||
yield headers
|
yield headers
|
||||||
|
|
||||||
full_fee_sum_cache = {
|
full_fee_sum_cache = {
|
||||||
@@ -503,8 +504,9 @@ class OrderListExporter(MultiSheetListExporter):
|
|||||||
headers.append(_('External customer ID'))
|
headers.append(_('External customer ID'))
|
||||||
headers.append(_('Payment providers'))
|
headers.append(_('Payment providers'))
|
||||||
|
|
||||||
# get meta_data labels from first cached event
|
if self.event_object_cache:
|
||||||
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
|
# get meta_data labels from first cached event if any
|
||||||
|
headers += next(iter(self.event_object_cache.values())).meta_data.keys()
|
||||||
yield headers
|
yield headers
|
||||||
|
|
||||||
yield self.ProgressSetTotal(total=qs.count())
|
yield self.ProgressSetTotal(total=qs.count())
|
||||||
@@ -707,9 +709,9 @@ class OrderListExporter(MultiSheetListExporter):
|
|||||||
_('Position order link')
|
_('Position order link')
|
||||||
]
|
]
|
||||||
|
|
||||||
# get meta_data labels from first cached event
|
|
||||||
meta_data_labels = next(iter(self.event_object_cache.values())).meta_data.keys()
|
|
||||||
if has_subevents:
|
if has_subevents:
|
||||||
|
# get meta_data labels from first cached event
|
||||||
|
meta_data_labels = next(iter(self.event_object_cache.values())).meta_data.keys()
|
||||||
headers += meta_data_labels
|
headers += meta_data_labels
|
||||||
yield headers
|
yield headers
|
||||||
|
|
||||||
|
|||||||
@@ -1415,6 +1415,7 @@ class BaseInvoiceAddressForm(forms.ModelForm):
|
|||||||
if not data.get(r):
|
if not data.get(r):
|
||||||
raise ValidationError({r: _("This field is required for the selected type of invoice transmission.")})
|
raise ValidationError({r: _("This field is required for the selected type of invoice transmission.")})
|
||||||
|
|
||||||
|
transmission_type.validate_invoice_address_data(data)
|
||||||
self.instance.transmission_type = transmission_type.identifier
|
self.instance.transmission_type = transmission_type.identifier
|
||||||
self.instance.transmission_info = transmission_type.form_data_to_transmission_info(data)
|
self.instance.transmission_info = transmission_type.form_data_to_transmission_info(data)
|
||||||
elif transmission_type.is_exclusive(self.event, data.get("country"), data.get("is_business")):
|
elif transmission_type.is_exclusive(self.event, data.get("country"), data.get("is_business")):
|
||||||
|
|||||||
@@ -33,8 +33,7 @@ from pretix.base.invoicing.transmission import (
|
|||||||
transmission_types,
|
transmission_types,
|
||||||
)
|
)
|
||||||
from pretix.base.models import Invoice, InvoiceAddress
|
from pretix.base.models import Invoice, InvoiceAddress
|
||||||
from pretix.base.services.mail import mail, render_mail
|
from pretix.base.services.mail import mail
|
||||||
from pretix.helpers.format import format_map
|
|
||||||
|
|
||||||
|
|
||||||
@transmission_types.new()
|
@transmission_types.new()
|
||||||
@@ -134,9 +133,7 @@ class EmailTransmissionProvider(TransmissionProvider):
|
|||||||
subject = invoice.order.event.settings.get('mail_subject_order_invoice', as_type=LazyI18nString)
|
subject = invoice.order.event.settings.get('mail_subject_order_invoice', as_type=LazyI18nString)
|
||||||
|
|
||||||
# Do not set to completed because that is done by the email sending task
|
# Do not set to completed because that is done by the email sending task
|
||||||
subject = format_map(subject, context)
|
outgoing_mail = mail(
|
||||||
email_content = render_mail(template, context)
|
|
||||||
mail(
|
|
||||||
[recipient],
|
[recipient],
|
||||||
subject,
|
subject,
|
||||||
template,
|
template,
|
||||||
@@ -151,19 +148,10 @@ class EmailTransmissionProvider(TransmissionProvider):
|
|||||||
plain_text_only=True,
|
plain_text_only=True,
|
||||||
no_order_links=True,
|
no_order_links=True,
|
||||||
)
|
)
|
||||||
invoice.order.log_action(
|
if outgoing_mail:
|
||||||
'pretix.event.order.email.invoice',
|
invoice.order.log_action(
|
||||||
user=None,
|
'pretix.event.order.email.invoice',
|
||||||
auth=None,
|
user=None,
|
||||||
data={
|
auth=None,
|
||||||
'subject': subject,
|
data=outgoing_mail.log_data()
|
||||||
'message': email_content,
|
)
|
||||||
'position': None,
|
|
||||||
'recipient': recipient,
|
|
||||||
'invoices': [invoice.pk],
|
|
||||||
'attach_tickets': False,
|
|
||||||
'attach_ical': False,
|
|
||||||
'attach_other_files': [],
|
|
||||||
'attach_cached_files': [],
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|||||||
@@ -148,6 +148,10 @@ class NumberedCanvas(Canvas):
|
|||||||
self.restoreState()
|
self.restoreState()
|
||||||
|
|
||||||
|
|
||||||
|
class InvoiceNotReadyException(Exception):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class BaseInvoiceRenderer:
|
class BaseInvoiceRenderer:
|
||||||
"""
|
"""
|
||||||
This is the base class for all invoice renderers.
|
This is the base class for all invoice renderers.
|
||||||
|
|||||||
@@ -204,6 +204,12 @@ class PeppolTransmissionType(TransmissionType):
|
|||||||
}
|
}
|
||||||
return base | {"transmission_peppol_participant_id"}
|
return base | {"transmission_peppol_participant_id"}
|
||||||
|
|
||||||
|
def validate_invoice_address_data(self, address_data: dict):
|
||||||
|
# Special case Belgium: If a Belgian business ID is used as Peppol ID, it should match the VAT ID
|
||||||
|
if address_data.get("transmission_peppol_participant_id").startswith("0208:") and address_data.get("vat_id"):
|
||||||
|
if address_data["vat_id"].removeprefix("BE") != address_data["transmission_peppol_participant_id"].removeprefix("0208:"):
|
||||||
|
raise ValidationError({"transmission_peppol_participant_id": _("The Peppol participant ID does not match your VAT ID.")})
|
||||||
|
|
||||||
def pdf_watermark(self) -> str:
|
def pdf_watermark(self) -> str:
|
||||||
return pgettext("peppol_invoice", "Visual copy")
|
return pgettext("peppol_invoice", "Visual copy")
|
||||||
|
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ from typing import Optional
|
|||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from django_countries.fields import Country
|
from django_countries.fields import Country
|
||||||
|
|
||||||
from pretix.base.models import Invoice, InvoiceAddress
|
from pretix.base.models import Invoice
|
||||||
from pretix.base.signals import EventPluginRegistry, Registry
|
from pretix.base.signals import EventPluginRegistry, Registry
|
||||||
|
|
||||||
|
|
||||||
@@ -89,7 +89,7 @@ class TransmissionType:
|
|||||||
def invoice_address_form_fields_visible(self, country: Country, is_business: bool) -> set:
|
def invoice_address_form_fields_visible(self, country: Country, is_business: bool) -> set:
|
||||||
return set(self.invoice_address_form_fields.keys())
|
return set(self.invoice_address_form_fields.keys())
|
||||||
|
|
||||||
def validate_address(self, ia: InvoiceAddress):
|
def validate_invoice_address_data(self, address_data: dict):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
|||||||
@@ -220,3 +220,20 @@ class OutgoingMail(models.Model):
|
|||||||
error_log_action_type = 'pretix.email.error'
|
error_log_action_type = 'pretix.email.error'
|
||||||
log_target = None
|
log_target = None
|
||||||
return log_target, error_log_action_type
|
return log_target, error_log_action_type
|
||||||
|
|
||||||
|
def log_data(self):
|
||||||
|
return {
|
||||||
|
"subject": self.subject,
|
||||||
|
"message": self.body_plain,
|
||||||
|
"to": self.to,
|
||||||
|
"cc": self.cc,
|
||||||
|
"bcc": self.bcc,
|
||||||
|
|
||||||
|
"invoices": [i.pk for i in self.should_attach_invoices.all()],
|
||||||
|
"attach_tickets": self.should_attach_tickets,
|
||||||
|
"attach_ical": self.should_attach_ical,
|
||||||
|
"attach_other_files": self.should_attach_other_files,
|
||||||
|
"attach_cached_files": [cf.filename for cf in self.should_attach_cached_files.all()],
|
||||||
|
|
||||||
|
"position": self.orderposition.positionid if self.orderposition else None,
|
||||||
|
}
|
||||||
|
|||||||
@@ -87,7 +87,6 @@ from pretix.base.timemachine import time_machine_now
|
|||||||
|
|
||||||
from ...helpers import OF_SELF
|
from ...helpers import OF_SELF
|
||||||
from ...helpers.countries import CachedCountries, FastCountryField
|
from ...helpers.countries import CachedCountries, FastCountryField
|
||||||
from ...helpers.format import FormattedString, format_map
|
|
||||||
from ...helpers.names import build_name
|
from ...helpers.names import build_name
|
||||||
from ...testutils.middleware import debugflags_var
|
from ...testutils.middleware import debugflags_var
|
||||||
from ._transactions import (
|
from ._transactions import (
|
||||||
@@ -1167,7 +1166,7 @@ class Order(LockModel, LoggedModel):
|
|||||||
only be attached for this position and child positions, the link will only point to the
|
only be attached for this position and child positions, the link will only point to the
|
||||||
position and the attendee email will be used if available.
|
position and the attendee email will be used if available.
|
||||||
"""
|
"""
|
||||||
from pretix.base.services.mail import mail, render_mail
|
from pretix.base.services.mail import mail
|
||||||
|
|
||||||
if not self.email and not (position and position.attendee_email):
|
if not self.email and not (position and position.attendee_email):
|
||||||
return
|
return
|
||||||
@@ -1177,32 +1176,20 @@ class Order(LockModel, LoggedModel):
|
|||||||
if position and position.attendee_email:
|
if position and position.attendee_email:
|
||||||
recipient = position.attendee_email
|
recipient = position.attendee_email
|
||||||
|
|
||||||
email_content = render_mail(template, context)
|
outgoing_mail = mail(
|
||||||
if not isinstance(subject, FormattedString):
|
|
||||||
subject = format_map(subject, context)
|
|
||||||
mail(
|
|
||||||
recipient, subject, template, context,
|
recipient, subject, template, context,
|
||||||
self.event, self.locale, self, headers=headers, sender=sender,
|
self.event, self.locale, self, headers=headers, sender=sender,
|
||||||
invoices=invoices, attach_tickets=attach_tickets,
|
invoices=invoices, attach_tickets=attach_tickets,
|
||||||
position=position, auto_email=auto_email, attach_ical=attach_ical,
|
position=position, auto_email=auto_email, attach_ical=attach_ical,
|
||||||
attach_other_files=attach_other_files, attach_cached_files=attach_cached_files,
|
attach_other_files=attach_other_files, attach_cached_files=attach_cached_files,
|
||||||
)
|
)
|
||||||
self.log_action(
|
if outgoing_mail:
|
||||||
log_entry_type,
|
self.log_action(
|
||||||
user=user,
|
log_entry_type,
|
||||||
auth=auth,
|
user=user,
|
||||||
data={
|
auth=auth,
|
||||||
'subject': subject,
|
data=outgoing_mail.log_data(),
|
||||||
'message': email_content,
|
)
|
||||||
'position': position.positionid if position else None,
|
|
||||||
'recipient': recipient,
|
|
||||||
'invoices': [i.pk for i in invoices] if invoices else [],
|
|
||||||
'attach_tickets': attach_tickets,
|
|
||||||
'attach_ical': attach_ical,
|
|
||||||
'attach_other_files': attach_other_files,
|
|
||||||
'attach_cached_files': [cf.filename for cf in attach_cached_files] if attach_cached_files else [],
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
def resend_link(self, user=None, auth=None):
|
def resend_link(self, user=None, auth=None):
|
||||||
with language(self.locale, self.event.settings.region):
|
with language(self.locale, self.event.settings.region):
|
||||||
@@ -2900,17 +2887,14 @@ class OrderPosition(AbstractPosition):
|
|||||||
:param attach_tickets: Attach tickets of this order, if they are existing and ready to download
|
:param attach_tickets: Attach tickets of this order, if they are existing and ready to download
|
||||||
:param attach_ical: Attach relevant ICS files
|
:param attach_ical: Attach relevant ICS files
|
||||||
"""
|
"""
|
||||||
from pretix.base.services.mail import mail, render_mail
|
from pretix.base.services.mail import mail
|
||||||
|
|
||||||
if not self.attendee_email:
|
if not self.attendee_email:
|
||||||
return
|
return
|
||||||
|
|
||||||
with language(self.order.locale, self.order.event.settings.region):
|
with language(self.order.locale, self.order.event.settings.region):
|
||||||
recipient = self.attendee_email
|
recipient = self.attendee_email
|
||||||
email_content = render_mail(template, context)
|
outgoing_mail = mail(
|
||||||
if not isinstance(subject, FormattedString):
|
|
||||||
subject = format_map(subject, context)
|
|
||||||
mail(
|
|
||||||
recipient, subject, template, context,
|
recipient, subject, template, context,
|
||||||
self.event, self.order.locale, order=self.order, headers=headers, sender=sender,
|
self.event, self.order.locale, order=self.order, headers=headers, sender=sender,
|
||||||
position=self,
|
position=self,
|
||||||
@@ -2919,21 +2903,13 @@ class OrderPosition(AbstractPosition):
|
|||||||
attach_ical=attach_ical,
|
attach_ical=attach_ical,
|
||||||
attach_other_files=attach_other_files,
|
attach_other_files=attach_other_files,
|
||||||
)
|
)
|
||||||
self.order.log_action(
|
if outgoing_mail:
|
||||||
log_entry_type,
|
self.order.log_action(
|
||||||
user=user,
|
log_entry_type,
|
||||||
auth=auth,
|
user=user,
|
||||||
data={
|
auth=auth,
|
||||||
'subject': subject,
|
data=outgoing_mail.log_data(),
|
||||||
'message': email_content,
|
)
|
||||||
'recipient': recipient,
|
|
||||||
'invoices': [i.pk for i in invoices] if invoices else [],
|
|
||||||
'attach_tickets': attach_tickets,
|
|
||||||
'attach_ical': attach_ical,
|
|
||||||
'attach_other_files': attach_other_files,
|
|
||||||
'attach_cached_files': [],
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
def resend_link(self, user=None, auth=None):
|
def resend_link(self, user=None, auth=None):
|
||||||
|
|
||||||
|
|||||||
@@ -34,10 +34,9 @@ from phonenumber_field.modelfields import PhoneNumberField
|
|||||||
from pretix.base.email import get_email_context
|
from pretix.base.email import get_email_context
|
||||||
from pretix.base.i18n import language
|
from pretix.base.i18n import language
|
||||||
from pretix.base.models import User, Voucher
|
from pretix.base.models import User, Voucher
|
||||||
from pretix.base.services.mail import mail, render_mail
|
from pretix.base.services.mail import mail
|
||||||
from pretix.helpers import OF_SELF
|
from pretix.helpers import OF_SELF
|
||||||
|
|
||||||
from ...helpers.format import format_map
|
|
||||||
from ...helpers.names import build_name
|
from ...helpers.names import build_name
|
||||||
from .base import LoggedModel
|
from .base import LoggedModel
|
||||||
from .event import Event, SubEvent
|
from .event import Event, SubEvent
|
||||||
@@ -181,10 +180,11 @@ class WaitingListEntry(LoggedModel):
|
|||||||
block_quota=True,
|
block_quota=True,
|
||||||
item_id=self.item_id,
|
item_id=self.item_id,
|
||||||
subevent_id=self.subevent_id,
|
subevent_id=self.subevent_id,
|
||||||
waitinglistentries__isnull=False
|
waitinglistentries__isnull=False,
|
||||||
|
seat__isnull=True
|
||||||
).aggregate(free=Sum(F('max_usages') - F('redeemed')))['free'] or 0
|
).aggregate(free=Sum(F('max_usages') - F('redeemed')))['free'] or 0
|
||||||
free_seats = num_free_seats_for_product - num_valid_vouchers_for_product
|
free_seats = num_free_seats_for_product - num_valid_vouchers_for_product
|
||||||
if not free_seats:
|
if free_seats < 1:
|
||||||
raise WaitingListException(_('No seat with this product is currently available.'))
|
raise WaitingListException(_('No seat with this product is currently available.'))
|
||||||
|
|
||||||
if '@' not in self.email:
|
if '@' not in self.email:
|
||||||
@@ -272,9 +272,7 @@ class WaitingListEntry(LoggedModel):
|
|||||||
with language(self.locale, self.event.settings.region):
|
with language(self.locale, self.event.settings.region):
|
||||||
recipient = self.email
|
recipient = self.email
|
||||||
|
|
||||||
email_content = render_mail(template, context)
|
outgoing_mail = mail(
|
||||||
subject = format_map(subject, context)
|
|
||||||
mail(
|
|
||||||
recipient, subject, template, context,
|
recipient, subject, template, context,
|
||||||
self.event,
|
self.event,
|
||||||
self.locale,
|
self.locale,
|
||||||
@@ -284,18 +282,13 @@ class WaitingListEntry(LoggedModel):
|
|||||||
attach_other_files=attach_other_files,
|
attach_other_files=attach_other_files,
|
||||||
attach_cached_files=attach_cached_files,
|
attach_cached_files=attach_cached_files,
|
||||||
)
|
)
|
||||||
self.log_action(
|
if outgoing_mail:
|
||||||
log_entry_type,
|
self.log_action(
|
||||||
user=user,
|
log_entry_type,
|
||||||
auth=auth,
|
user=user,
|
||||||
data={
|
auth=auth,
|
||||||
'subject': subject,
|
data=outgoing_mail.log_data(),
|
||||||
'message': email_content,
|
)
|
||||||
'recipient': recipient,
|
|
||||||
'attach_other_files': attach_other_files,
|
|
||||||
'attach_cached_files': [cf.filename for cf in attach_cached_files] if attach_cached_files else [],
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def clean_itemvar(event, item, variation):
|
def clean_itemvar(event, item, variation):
|
||||||
|
|||||||
@@ -1295,6 +1295,7 @@ class ManualPayment(BasePaymentProvider):
|
|||||||
|
|
||||||
def format_map(self, order, payment):
|
def format_map(self, order, payment):
|
||||||
return {
|
return {
|
||||||
|
# Possible placeholder injection, we should make sure to never include user-controlled variables here
|
||||||
'order': order.code,
|
'order': order.code,
|
||||||
'amount': payment.amount,
|
'amount': payment.amount,
|
||||||
'currency': self.event.currency,
|
'currency': self.event.currency,
|
||||||
|
|||||||
@@ -49,14 +49,39 @@ class PluginType(Enum):
|
|||||||
EXPORT = 4
|
EXPORT = 4
|
||||||
|
|
||||||
|
|
||||||
|
def plugin_is_available(meta, event=None, organizer=None):
|
||||||
|
if not hasattr(meta.app, 'is_available'):
|
||||||
|
return True
|
||||||
|
|
||||||
|
level = getattr(meta, "level", PLUGIN_LEVEL_EVENT)
|
||||||
|
if level == PLUGIN_LEVEL_EVENT:
|
||||||
|
if event:
|
||||||
|
return meta.app.is_available(event)
|
||||||
|
elif organizer:
|
||||||
|
if not hasattr(organizer, '_plugin_availability_fallback_event'):
|
||||||
|
with scope(organizer=organizer):
|
||||||
|
setattr(organizer, '_plugin_availability_fallback_event', organizer.events.first())
|
||||||
|
return (
|
||||||
|
organizer._plugin_availability_fallback_event
|
||||||
|
and meta.app.is_available(organizer._plugin_availability_fallback_event)
|
||||||
|
)
|
||||||
|
elif level == PLUGIN_LEVEL_ORGANIZER:
|
||||||
|
if organizer:
|
||||||
|
return meta.app.is_available(organizer)
|
||||||
|
elif event:
|
||||||
|
return meta.app.is_available(event.organizer)
|
||||||
|
elif level == PLUGIN_LEVEL_EVENT_ORGANIZER_HYBRID and (event or organizer):
|
||||||
|
return meta.app.is_available(event or organizer)
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
|
||||||
def get_all_plugins(*, event=None, organizer=None) -> List[type]:
|
def get_all_plugins(*, event=None, organizer=None) -> List[type]:
|
||||||
"""
|
"""
|
||||||
Returns the PretixPluginMeta classes of all plugins found in the installed Django apps.
|
Returns the PretixPluginMeta classes of all plugins found in the installed Django apps.
|
||||||
"""
|
"""
|
||||||
assert not event or not organizer
|
assert not event or not organizer
|
||||||
plugins = []
|
plugins = []
|
||||||
event_fallback = None
|
|
||||||
event_fallback_used = False
|
|
||||||
for app in apps.get_app_configs():
|
for app in apps.get_app_configs():
|
||||||
if hasattr(app, 'PretixPluginMeta'):
|
if hasattr(app, 'PretixPluginMeta'):
|
||||||
meta = app.PretixPluginMeta
|
meta = app.PretixPluginMeta
|
||||||
@@ -65,28 +90,8 @@ def get_all_plugins(*, event=None, organizer=None) -> List[type]:
|
|||||||
if app.name in settings.PRETIX_PLUGINS_EXCLUDE:
|
if app.name in settings.PRETIX_PLUGINS_EXCLUDE:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
level = getattr(meta, "level", PLUGIN_LEVEL_EVENT)
|
if not plugin_is_available(meta, event, organizer):
|
||||||
if level == PLUGIN_LEVEL_EVENT:
|
continue
|
||||||
if event and hasattr(app, 'is_available'):
|
|
||||||
if not app.is_available(event):
|
|
||||||
continue
|
|
||||||
elif organizer and hasattr(app, 'is_available'):
|
|
||||||
if not event_fallback_used:
|
|
||||||
with scope(organizer=organizer):
|
|
||||||
event_fallback = organizer.events.first()
|
|
||||||
event_fallback_used = True
|
|
||||||
if not event_fallback or not app.is_available(event_fallback):
|
|
||||||
continue
|
|
||||||
elif level == PLUGIN_LEVEL_ORGANIZER:
|
|
||||||
if organizer and hasattr(app, 'is_available'):
|
|
||||||
if not app.is_available(organizer):
|
|
||||||
continue
|
|
||||||
elif event and hasattr(app, 'is_available'):
|
|
||||||
if not app.is_available(event.organizer):
|
|
||||||
continue
|
|
||||||
elif level == PLUGIN_LEVEL_EVENT_ORGANIZER_HYBRID and (event or organizer) and hasattr(app, 'is_available'):
|
|
||||||
if not app.is_available(event or organizer):
|
|
||||||
continue
|
|
||||||
|
|
||||||
plugins.append(meta)
|
plugins.append(meta)
|
||||||
return sorted(
|
return sorted(
|
||||||
|
|||||||
@@ -45,7 +45,6 @@ from pretix.base.services.tax import split_fee_for_taxes
|
|||||||
from pretix.base.templatetags.money import money_filter
|
from pretix.base.templatetags.money import money_filter
|
||||||
from pretix.celery_app import app
|
from pretix.celery_app import app
|
||||||
from pretix.helpers import OF_SELF
|
from pretix.helpers import OF_SELF
|
||||||
from pretix.helpers.format import format_map
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@@ -55,7 +54,7 @@ def _send_wle_mail(wle: WaitingListEntry, subject: LazyI18nString, message: Lazy
|
|||||||
email_context = get_email_context(event_or_subevent=subevent or wle.event, event=wle.event)
|
email_context = get_email_context(event_or_subevent=subevent or wle.event, event=wle.event)
|
||||||
mail(
|
mail(
|
||||||
wle.email,
|
wle.email,
|
||||||
format_map(subject, email_context),
|
str(subject),
|
||||||
message,
|
message,
|
||||||
email_context,
|
email_context,
|
||||||
wle.event,
|
wle.event,
|
||||||
@@ -73,9 +72,8 @@ def _send_mail(order: Order, subject: LazyI18nString, message: LazyI18nString, s
|
|||||||
|
|
||||||
email_context = get_email_context(event_or_subevent=subevent or order.event, refund_amount=refund_amount,
|
email_context = get_email_context(event_or_subevent=subevent or order.event, refund_amount=refund_amount,
|
||||||
order=order, position_or_address=ia, event=order.event)
|
order=order, position_or_address=ia, event=order.event)
|
||||||
real_subject = format_map(subject, email_context)
|
|
||||||
order.send_mail(
|
order.send_mail(
|
||||||
real_subject, message, email_context,
|
subject, message, email_context,
|
||||||
'pretix.event.order.email.event_canceled',
|
'pretix.event.order.email.event_canceled',
|
||||||
user,
|
user,
|
||||||
)
|
)
|
||||||
@@ -85,14 +83,13 @@ def _send_mail(order: Order, subject: LazyI18nString, message: LazyI18nString, s
|
|||||||
continue
|
continue
|
||||||
|
|
||||||
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
||||||
real_subject = format_map(subject, email_context)
|
|
||||||
email_context = get_email_context(event_or_subevent=p.subevent or order.event,
|
email_context = get_email_context(event_or_subevent=p.subevent or order.event,
|
||||||
event=order.event,
|
event=order.event,
|
||||||
refund_amount=refund_amount,
|
refund_amount=refund_amount,
|
||||||
position_or_address=p,
|
position_or_address=p,
|
||||||
order=order, position=p)
|
order=order, position=p)
|
||||||
order.send_mail(
|
order.send_mail(
|
||||||
real_subject, message, email_context,
|
subject, message, email_context,
|
||||||
'pretix.event.order.email.event_canceled',
|
'pretix.event.order.email.event_canceled',
|
||||||
position=p,
|
position=p,
|
||||||
user=user
|
user=user
|
||||||
|
|||||||
@@ -334,8 +334,7 @@ def _check_position_constraints(
|
|||||||
raise CartPositionError(error_messages['voucher_invalid_subevent'])
|
raise CartPositionError(error_messages['voucher_invalid_subevent'])
|
||||||
|
|
||||||
# Voucher expired
|
# Voucher expired
|
||||||
# (checked using real_now_dt as vouchers influence quota calculations)
|
if voucher and voucher.valid_until and voucher.valid_until < time_machine_now_dt:
|
||||||
if voucher and voucher.valid_until and voucher.valid_until < real_now_dt:
|
|
||||||
raise CartPositionError(error_messages['voucher_expired'])
|
raise CartPositionError(error_messages['voucher_expired'])
|
||||||
|
|
||||||
# Subevent has been disabled
|
# Subevent has been disabled
|
||||||
|
|||||||
@@ -51,6 +51,7 @@ from django_scopes import scope, scopes_disabled
|
|||||||
from i18nfield.strings import LazyI18nString
|
from i18nfield.strings import LazyI18nString
|
||||||
|
|
||||||
from pretix.base.i18n import language
|
from pretix.base.i18n import language
|
||||||
|
from pretix.base.invoicing.pdf import InvoiceNotReadyException
|
||||||
from pretix.base.invoicing.transmission import (
|
from pretix.base.invoicing.transmission import (
|
||||||
get_transmission_types, transmission_providers,
|
get_transmission_types, transmission_providers,
|
||||||
)
|
)
|
||||||
@@ -504,7 +505,7 @@ def generate_invoice(order: Order, trigger_pdf=True):
|
|||||||
return invoice
|
return invoice
|
||||||
|
|
||||||
|
|
||||||
@app.task(base=TransactionAwareTask)
|
@app.task(base=TransactionAwareTask, throws=(InvoiceNotReadyException,))
|
||||||
def invoice_pdf_task(invoice: int):
|
def invoice_pdf_task(invoice: int):
|
||||||
with scopes_disabled():
|
with scopes_disabled():
|
||||||
i = Invoice.objects.get(pk=invoice)
|
i = Invoice.objects.get(pk=invoice)
|
||||||
|
|||||||
@@ -149,13 +149,13 @@ def prefix_subject(settings_holder, subject, highlight=False):
|
|||||||
return subject
|
return subject
|
||||||
|
|
||||||
|
|
||||||
def mail(email: Union[str, Sequence[str]], subject: str, template: Union[str, LazyI18nString],
|
def mail(email: Union[str, Sequence[str]], subject: Union[str, FormattedString], template: Union[str, LazyI18nString],
|
||||||
context: Dict[str, Any] = None, event: Event = None, locale: str = None, order: Order = None,
|
context: Dict[str, Any] = None, event: Event = None, locale: str = None, order: Order = None,
|
||||||
position: OrderPosition = None, *, headers: dict = None, sender: str = None, organizer: Organizer = None,
|
position: OrderPosition = None, *, headers: dict = None, sender: str = None, organizer: Organizer = None,
|
||||||
customer: Customer = None, invoices: Sequence = None, attach_tickets=False, auto_email=True, user=None,
|
customer: Customer = None, invoices: Sequence = None, attach_tickets=False, auto_email=True, user=None,
|
||||||
attach_ical=False, attach_cached_files: Sequence = None, attach_other_files: list=None,
|
attach_ical=False, attach_cached_files: Sequence = None, attach_other_files: list=None,
|
||||||
plain_text_only=False, no_order_links=False, cc: Sequence[str]=None, bcc: Sequence[str]=None,
|
plain_text_only=False, no_order_links=False, cc: Sequence[str]=None, bcc: Sequence[str]=None,
|
||||||
sensitive: bool=False):
|
sensitive: bool=False) -> Optional[OutgoingMail]:
|
||||||
"""
|
"""
|
||||||
Sends out an email to a user. The mail will be sent synchronously or asynchronously depending on the installation.
|
Sends out an email to a user. The mail will be sent synchronously or asynchronously depending on the installation.
|
||||||
|
|
||||||
@@ -335,14 +335,26 @@ def mail(email: Union[str, Sequence[str]], subject: str, template: Union[str, La
|
|||||||
should_attach_other_files=attach_other_files or [],
|
should_attach_other_files=attach_other_files or [],
|
||||||
sensitive=sensitive,
|
sensitive=sensitive,
|
||||||
)
|
)
|
||||||
|
m._prefetched_objects_cache = {}
|
||||||
if invoices and not position:
|
if invoices and not position:
|
||||||
m.should_attach_invoices.add(*invoices)
|
m.should_attach_invoices.add(*invoices)
|
||||||
|
# Hack: For logging, we'll later make a `should_attach_invoices.all()` call. We can prevent a useless
|
||||||
|
# DB query by filling the cache
|
||||||
|
m._prefetched_objects_cache[m.should_attach_invoices.prefetch_cache_name] = invoices
|
||||||
|
else:
|
||||||
|
m._prefetched_objects_cache[m.should_attach_invoices.prefetch_cache_name] = Invoice.objects.none()
|
||||||
if attach_cached_files:
|
if attach_cached_files:
|
||||||
|
cf_list = []
|
||||||
for cf in attach_cached_files:
|
for cf in attach_cached_files:
|
||||||
if not isinstance(cf, CachedFile):
|
if not isinstance(cf, CachedFile):
|
||||||
m.should_attach_cached_files.add(CachedFile.objects.get(pk=cf))
|
cf = CachedFile.objects.get(pk=cf)
|
||||||
else:
|
m.should_attach_cached_files.add(cf)
|
||||||
m.should_attach_cached_files.add(cf)
|
cf_list.append(cf)
|
||||||
|
# Hack: For logging, we'll later make a `should_attach_cached_files.all()` call. We can prevent a useless
|
||||||
|
# DB query by filling the cache
|
||||||
|
m._prefetched_objects_cache[m.should_attach_cached_files.prefetch_cache_name] = cf_list
|
||||||
|
else:
|
||||||
|
m._prefetched_objects_cache[m.should_attach_cached_files.prefetch_cache_name] = CachedFile.objects.none()
|
||||||
|
|
||||||
send_task = mail_send_task.si(
|
send_task = mail_send_task.si(
|
||||||
outgoing_mail=m.id
|
outgoing_mail=m.id
|
||||||
@@ -364,6 +376,8 @@ def mail(email: Union[str, Sequence[str]], subject: str, template: Union[str, La
|
|||||||
lambda: chain(*task_chain).apply_async()
|
lambda: chain(*task_chain).apply_async()
|
||||||
)
|
)
|
||||||
|
|
||||||
|
return m
|
||||||
|
|
||||||
|
|
||||||
class CustomEmail(EmailMultiAlternatives):
|
class CustomEmail(EmailMultiAlternatives):
|
||||||
def _create_mime_attachment(self, content, mimetype):
|
def _create_mime_attachment(self, content, mimetype):
|
||||||
@@ -409,6 +423,18 @@ def mail_send_task(self, **kwargs) -> bool:
|
|||||||
outgoing_mail.inflight_since = now()
|
outgoing_mail.inflight_since = now()
|
||||||
outgoing_mail.save(update_fields=["status", "inflight_since"])
|
outgoing_mail.save(update_fields=["status", "inflight_since"])
|
||||||
|
|
||||||
|
# Performance optimization, saves database queries later on if we resolve the known relationships
|
||||||
|
if outgoing_mail.event_id:
|
||||||
|
assert outgoing_mail.event.organizer_id == outgoing_mail.organizer.pk
|
||||||
|
outgoing_mail.event.organizer = outgoing_mail.organizer
|
||||||
|
if outgoing_mail.order_id:
|
||||||
|
assert outgoing_mail.order.event_id == outgoing_mail.event_id
|
||||||
|
outgoing_mail.order.event = outgoing_mail.event
|
||||||
|
outgoing_mail.order.organizer = outgoing_mail.organizer
|
||||||
|
if outgoing_mail.orderposition_id:
|
||||||
|
assert outgoing_mail.orderposition.order_id == outgoing_mail.order_id
|
||||||
|
outgoing_mail.orderposition.order = outgoing_mail.order
|
||||||
|
|
||||||
headers = dict(outgoing_mail.headers)
|
headers = dict(outgoing_mail.headers)
|
||||||
headers.setdefault('X-PX-Correlation', str(outgoing_mail.guid))
|
headers.setdefault('X-PX-Correlation', str(outgoing_mail.guid))
|
||||||
email = CustomEmail(
|
email = CustomEmail(
|
||||||
|
|||||||
@@ -1799,8 +1799,6 @@ class OrderChangeManager:
|
|||||||
tax_rule = tax_rules.get(pos.pk, pos.tax_rule)
|
tax_rule = tax_rules.get(pos.pk, pos.tax_rule)
|
||||||
if not tax_rule:
|
if not tax_rule:
|
||||||
continue
|
continue
|
||||||
if not pos.price:
|
|
||||||
continue
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
new_rate = tax_rule.tax_rate_for(ia)
|
new_rate = tax_rule.tax_rate_for(ia)
|
||||||
@@ -1817,7 +1815,9 @@ class OrderChangeManager:
|
|||||||
override_tax_rate=new_rate, override_tax_code=new_code)
|
override_tax_rate=new_rate, override_tax_code=new_code)
|
||||||
self._totaldiff_guesstimate += new_tax.gross - pos.price
|
self._totaldiff_guesstimate += new_tax.gross - pos.price
|
||||||
self._operations.append(self.PriceOperation(pos, new_tax, new_tax.gross - pos.price))
|
self._operations.append(self.PriceOperation(pos, new_tax, new_tax.gross - pos.price))
|
||||||
self._invoice_dirty = True
|
if pos.price:
|
||||||
|
# We do not consider the invoice dirty if only 0€-valued taxes are changed
|
||||||
|
self._invoice_dirty = True
|
||||||
|
|
||||||
def cancel_fee(self, fee: OrderFee):
|
def cancel_fee(self, fee: OrderFee):
|
||||||
self._totaldiff_guesstimate -= fee.value
|
self._totaldiff_guesstimate -= fee.value
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ import logging
|
|||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
from decimal import Decimal
|
from decimal import Decimal
|
||||||
|
|
||||||
|
from django.db.models import Prefetch, prefetch_related_objects
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
from django.utils.formats import date_format
|
from django.utils.formats import date_format
|
||||||
from django.utils.html import escape, mark_safe
|
from django.utils.html import escape, mark_safe
|
||||||
@@ -35,6 +36,7 @@ from pretix.base.forms.widgets import format_placeholders_help_text
|
|||||||
from pretix.base.i18n import (
|
from pretix.base.i18n import (
|
||||||
LazyCurrencyNumber, LazyDate, LazyExpiresDate, LazyNumber,
|
LazyCurrencyNumber, LazyDate, LazyExpiresDate, LazyNumber,
|
||||||
)
|
)
|
||||||
|
from pretix.base.models import EventMetaValue
|
||||||
from pretix.base.reldate import RelativeDateWrapper
|
from pretix.base.reldate import RelativeDateWrapper
|
||||||
from pretix.base.settings import PERSON_NAME_SCHEMES, get_name_parts_localized
|
from pretix.base.settings import PERSON_NAME_SCHEMES, get_name_parts_localized
|
||||||
from pretix.base.signals import (
|
from pretix.base.signals import (
|
||||||
@@ -752,6 +754,11 @@ def base_placeholders(sender, **kwargs):
|
|||||||
name_scheme['sample'][f]
|
name_scheme['sample'][f]
|
||||||
))
|
))
|
||||||
|
|
||||||
|
prefetch_related_objects(
|
||||||
|
[sender],
|
||||||
|
Prefetch('meta_values', queryset=EventMetaValue.objects.select_related("property"), to_attr="meta_values_cached")
|
||||||
|
)
|
||||||
|
prefetch_related_objects([sender.organizer], Prefetch('meta_properties'))
|
||||||
for k, v in sender.meta_data.items():
|
for k, v in sender.meta_data.items():
|
||||||
ph.append(MarkdownTextPlaceholder(
|
ph.append(MarkdownTextPlaceholder(
|
||||||
'meta_%s' % k, ['event'], lambda event, k=k: event.meta_data[k],
|
'meta_%s' % k, ['event'], lambda event, k=k: event.meta_data[k],
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ def vouchers_send(event: Event, vouchers: list, subject: str, message: str, reci
|
|||||||
with language(event.settings.locale):
|
with language(event.settings.locale):
|
||||||
email_context = get_email_context(event=event, name=r.get('name') or '',
|
email_context = get_email_context(event=event, name=r.get('name') or '',
|
||||||
voucher_list=[v.code for v in voucher_list])
|
voucher_list=[v.code for v in voucher_list])
|
||||||
mail(
|
outgoing_mail = mail(
|
||||||
r['email'],
|
r['email'],
|
||||||
subject,
|
subject,
|
||||||
LazyI18nString(message),
|
LazyI18nString(message),
|
||||||
@@ -60,8 +60,8 @@ def vouchers_send(event: Event, vouchers: list, subject: str, message: str, reci
|
|||||||
data={
|
data={
|
||||||
'recipient': r['email'],
|
'recipient': r['email'],
|
||||||
'name': r.get('name'),
|
'name': r.get('name'),
|
||||||
'subject': subject,
|
'subject': outgoing_mail.subject,
|
||||||
'message': message,
|
'message': outgoing_mail.body_plain,
|
||||||
},
|
},
|
||||||
save=False
|
save=False
|
||||||
))
|
))
|
||||||
|
|||||||
@@ -363,7 +363,7 @@ class EmailAddressShredder(BaseDataShredder):
|
|||||||
le.save(update_fields=['data', 'shredded'])
|
le.save(update_fields=['data', 'shredded'])
|
||||||
else:
|
else:
|
||||||
shred_log_fields(le, banlist=[
|
shred_log_fields(le, banlist=[
|
||||||
'recipient', 'message', 'subject', 'full_mail', 'old_email', 'new_email'
|
'recipient', 'message', 'subject', 'full_mail', 'old_email', 'new_email', 'bcc', 'cc',
|
||||||
])
|
])
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -12,6 +12,9 @@
|
|||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<link rel="icon" href="{% static "pretixbase/img/favicon.ico" %}">
|
<link rel="icon" href="{% static "pretixbase/img/favicon.ico" %}">
|
||||||
{% block custom_header %}{% endblock %}
|
{% block custom_header %}{% endblock %}
|
||||||
|
{% if css_theme %}
|
||||||
|
<link rel="stylesheet" type="text/css" href="{{ css_theme }}" />
|
||||||
|
{% endif %}
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div class="container">
|
<div class="container">
|
||||||
|
|||||||
@@ -423,7 +423,7 @@ def resolve_timeframe_to_dates_inclusive(ref_dt, frame, timezone) -> Tuple[Optio
|
|||||||
raise ValueError(f"Invalid timeframe '{frame}'")
|
raise ValueError(f"Invalid timeframe '{frame}'")
|
||||||
|
|
||||||
|
|
||||||
def resolve_timeframe_to_datetime_start_inclusive_end_exclusive(ref_dt, frame, timezone) -> Tuple[Optional[date], Optional[date]]:
|
def resolve_timeframe_to_datetime_start_inclusive_end_exclusive(ref_dt, frame, timezone) -> Tuple[Optional[datetime], Optional[datetime]]:
|
||||||
"""
|
"""
|
||||||
Given a serialized timeframe, evaluate it relative to `ref_dt` and return a tuple of datetimes
|
Given a serialized timeframe, evaluate it relative to `ref_dt` and return a tuple of datetimes
|
||||||
where the first element ist the first possible datetime within the timeframe and the second
|
where the first element ist the first possible datetime within the timeframe and the second
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% load urlreplace %}
|
{% load urlreplace %}
|
||||||
{% load bootstrap3 %}
|
{% load bootstrap3 %}
|
||||||
|
{% load static %}
|
||||||
{% block title %}{% trans "Events" %}{% endblock %}
|
{% block title %}{% trans "Events" %}{% endblock %}
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1>{% trans "Events" %}</h1>
|
<h1>{% trans "Events" %}</h1>
|
||||||
@@ -74,6 +75,7 @@
|
|||||||
<a href="?{% url_replace request 'ordering' 'organizer' %}"><i class="fa fa-caret-up"></i></a>
|
<a href="?{% url_replace request 'ordering' 'organizer' %}"><i class="fa fa-caret-up"></i></a>
|
||||||
</th>
|
</th>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
|
<th>{% trans "Sales channels" %}</th>
|
||||||
<th>
|
<th>
|
||||||
{% trans "Start date" %}
|
{% trans "Start date" %}
|
||||||
<a href="?{% url_replace request 'ordering' '-date_from' %}"><i class="fa fa-caret-down"></i></a>
|
<a href="?{% url_replace request 'ordering' '-date_from' %}"><i class="fa fa-caret-down"></i></a>
|
||||||
@@ -108,6 +110,21 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</td>
|
</td>
|
||||||
{% if not hide_orga %}<td>{{ e.organizer }}</td>{% endif %}
|
{% if not hide_orga %}<td>{{ e.organizer }}</td>{% endif %}
|
||||||
|
<td>
|
||||||
|
{% for c in e.organizer.sales_channels.all %}
|
||||||
|
{% if e.all_sales_channels or c in e.limit_sales_channels.all %}
|
||||||
|
{% if "." in c.icon %}
|
||||||
|
<img src="{% static c.icon %}" class="fa-like-image"
|
||||||
|
data-toggle="tooltip" title="{{ c.label }}">
|
||||||
|
{% else %}
|
||||||
|
<span class="fa fa-fw fa-{{ c.icon }} text-muted"
|
||||||
|
data-toggle="tooltip" title="{{ c.label }}"></span>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<span class="fa fa-fw"></span>
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
<td class="event-date-col">
|
<td class="event-date-col">
|
||||||
{% if e.has_subevents %}
|
{% if e.has_subevents %}
|
||||||
<span class="fa fa-fw- fa-calendar"></span>
|
<span class="fa fa-fw- fa-calendar"></span>
|
||||||
|
|||||||
@@ -24,7 +24,9 @@
|
|||||||
{% if log.display %}
|
{% if log.display %}
|
||||||
<br/><span class="fa fa-fw fa-comment-o"></span> {{ log.display }}
|
<br/><span class="fa fa-fw fa-comment-o"></span> {{ log.display }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if log.parsed_data.recipient %}
|
{% if log.parsed_data.to %}
|
||||||
|
<br/><span class="fa fa-fw fa-envelope-o"></span> {{ log.parsed_data.to|join:", " }}
|
||||||
|
{% elif log.parsed_data.recipient %} {# legacy #}
|
||||||
<br/><span class="fa fa-fw fa-envelope-o"></span> {{ log.parsed_data.recipient }}
|
<br/><span class="fa fa-fw fa-envelope-o"></span> {{ log.parsed_data.recipient }}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
{% extends "pretixcontrol/organizers/base.html" %}
|
{% extends "pretixcontrol/organizers/base.html" %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
{% load bootstrap3 %}
|
{% load bootstrap3 %}
|
||||||
|
{% load static %}
|
||||||
{% block inner %}
|
{% block inner %}
|
||||||
<h1>
|
<h1>
|
||||||
{% blocktrans with name=request.organizer.name %}Organizer: {{ name }}{% endblocktrans %}
|
{% blocktrans with name=request.organizer.name %}Organizer: {{ name }}{% endblocktrans %}
|
||||||
@@ -62,6 +63,7 @@
|
|||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>{% trans "Event name" %}</th>
|
<th>{% trans "Event name" %}</th>
|
||||||
|
<th>{% trans "Sales channels" %}</th>
|
||||||
<th>
|
<th>
|
||||||
{% trans "Start date" %}
|
{% trans "Start date" %}
|
||||||
/
|
/
|
||||||
@@ -77,10 +79,30 @@
|
|||||||
<td>
|
<td>
|
||||||
<strong><a
|
<strong><a
|
||||||
href="{% url "control:event.index" organizer=e.organizer.slug event=e.slug %}">{{ e.name }}</a></strong>
|
href="{% url "control:event.index" organizer=e.organizer.slug event=e.slug %}">{{ e.name }}</a></strong>
|
||||||
<br><small>{{ e.slug }}</small>
|
<br>
|
||||||
{% for k, v in e.meta_data.items %}
|
<small>
|
||||||
{% if v %}
|
{{ e.slug }}
|
||||||
<small class="text-muted">· {{ k }}: {{ v }}</small>
|
</small>
|
||||||
|
<small class="text-muted">
|
||||||
|
{% for k, v in e.meta_data.items %}
|
||||||
|
{% if v %}
|
||||||
|
· {{ k }}: {{ v }}
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</small>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{% for c in sales_channels %}
|
||||||
|
{% if e.all_sales_channels or c in e.limit_sales_channels.all %}
|
||||||
|
{% if "." in c.icon %}
|
||||||
|
<img src="{% static c.icon %}" class="fa-like-image"
|
||||||
|
data-toggle="tooltip" title="{{ c.label }}">
|
||||||
|
{% else %}
|
||||||
|
<span class="fa fa-fw fa-{{ c.icon }} text-muted"
|
||||||
|
data-toggle="tooltip" title="{{ c.label }}"></span>
|
||||||
|
{% endif %}
|
||||||
|
{% else %}
|
||||||
|
<span class="fa fa-fw"></span>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</td>
|
</td>
|
||||||
|
|||||||
@@ -264,12 +264,17 @@
|
|||||||
The paper size will match the PDF.
|
The paper size will match the PDF.
|
||||||
{% endblocktrans %}
|
{% endblocktrans %}
|
||||||
</p>
|
</p>
|
||||||
<p>
|
<p class="text-center">
|
||||||
<span class="btn btn-default fileinput-button background-button btn-block">
|
<span class="btn btn-default fileinput-button background-button btn-block">
|
||||||
<i class="fa fa-upload"></i>
|
<i class="fa fa-upload"></i>
|
||||||
<span>{% trans "Upload PDF as background" %}</span>
|
<span>{% trans "Upload PDF as background" %}</span>
|
||||||
<input id="fileupload" type="file" name="background" accept="application/pdf">
|
<input id="fileupload" type="file" name="background" accept="application/pdf">
|
||||||
</span>
|
</span>
|
||||||
|
<small class="text-muted">
|
||||||
|
{% blocktrans trimmed with size=maxfilesize|filesizeformat %}
|
||||||
|
max. {{ size }}, smaller is better
|
||||||
|
{% endblocktrans %}
|
||||||
|
</small>
|
||||||
</p>
|
</p>
|
||||||
<p class="text-center">
|
<p class="text-center">
|
||||||
<a class="btn btn-link background-download-button" href="{{ pdf }}" target="_blank">
|
<a class="btn btn-link background-download-button" href="{{ pdf }}" target="_blank">
|
||||||
|
|||||||
@@ -67,7 +67,12 @@ class EventList(PaginationMixin, ListView):
|
|||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
qs = self.request.user.get_events_with_any_permission(self.request).prefetch_related(
|
qs = self.request.user.get_events_with_any_permission(self.request).prefetch_related(
|
||||||
'organizer', '_settings_objects', 'organizer___settings_objects', 'organizer__meta_properties',
|
'organizer',
|
||||||
|
'organizer__sales_channels',
|
||||||
|
'_settings_objects',
|
||||||
|
'organizer___settings_objects',
|
||||||
|
'organizer__meta_properties',
|
||||||
|
'limit_sales_channels',
|
||||||
Prefetch(
|
Prefetch(
|
||||||
'meta_values',
|
'meta_values',
|
||||||
EventMetaValue.objects.select_related('property'),
|
EventMetaValue.objects.select_related('property'),
|
||||||
|
|||||||
@@ -2421,9 +2421,9 @@ class OrderSendMail(EventPermissionRequiredMixin, OrderViewMixin, FormView):
|
|||||||
with language(order.locale, self.request.event.settings.region):
|
with language(order.locale, self.request.event.settings.region):
|
||||||
email_context = get_email_context(event=order.event, order=order)
|
email_context = get_email_context(event=order.event, order=order)
|
||||||
email_template = LazyI18nString(form.cleaned_data['message'])
|
email_template = LazyI18nString(form.cleaned_data['message'])
|
||||||
email_subject = format_map(str(form.cleaned_data['subject']), email_context)
|
|
||||||
email_content = render_mail(email_template, email_context)
|
|
||||||
if self.request.POST.get('action') == 'preview':
|
if self.request.POST.get('action') == 'preview':
|
||||||
|
email_subject = format_map(form.cleaned_data['subject'], email_context)
|
||||||
|
email_content = render_mail(email_template, email_context)
|
||||||
self.preview_output = {
|
self.preview_output = {
|
||||||
'subject': mark_safe(_('Subject: {subject}').format(
|
'subject': mark_safe(_('Subject: {subject}').format(
|
||||||
subject=prefix_subject(order.event, escape(email_subject), highlight=True)
|
subject=prefix_subject(order.event, escape(email_subject), highlight=True)
|
||||||
@@ -2485,9 +2485,9 @@ class OrderPositionSendMail(OrderSendMail):
|
|||||||
with language(position.order.locale, self.request.event.settings.region):
|
with language(position.order.locale, self.request.event.settings.region):
|
||||||
email_context = get_email_context(event=position.order.event, order=position.order, position=position)
|
email_context = get_email_context(event=position.order.event, order=position.order, position=position)
|
||||||
email_template = LazyI18nString(form.cleaned_data['message'])
|
email_template = LazyI18nString(form.cleaned_data['message'])
|
||||||
email_subject = format_map(str(form.cleaned_data['subject']), email_context)
|
|
||||||
email_content = render_mail(email_template, email_context)
|
|
||||||
if self.request.POST.get('action') == 'preview':
|
if self.request.POST.get('action') == 'preview':
|
||||||
|
email_subject = format_map(str(form.cleaned_data['subject']), email_context)
|
||||||
|
email_content = render_mail(email_template, email_context)
|
||||||
self.preview_output = {
|
self.preview_output = {
|
||||||
'subject': mark_safe(_('Subject: {subject}').format(
|
'subject': mark_safe(_('Subject: {subject}').format(
|
||||||
subject=prefix_subject(position.order.event, escape(email_subject), highlight=True))
|
subject=prefix_subject(position.order.event, escape(email_subject), highlight=True))
|
||||||
|
|||||||
@@ -100,7 +100,7 @@ from pretix.base.models.organizer import SalesChannel, TeamAPIToken
|
|||||||
from pretix.base.payment import PaymentException
|
from pretix.base.payment import PaymentException
|
||||||
from pretix.base.plugins import (
|
from pretix.base.plugins import (
|
||||||
PLUGIN_LEVEL_EVENT, PLUGIN_LEVEL_EVENT_ORGANIZER_HYBRID,
|
PLUGIN_LEVEL_EVENT, PLUGIN_LEVEL_EVENT_ORGANIZER_HYBRID,
|
||||||
PLUGIN_LEVEL_ORGANIZER,
|
PLUGIN_LEVEL_ORGANIZER, plugin_is_available,
|
||||||
)
|
)
|
||||||
from pretix.base.services.export import multiexport, scheduled_organizer_export
|
from pretix.base.services.export import multiexport, scheduled_organizer_export
|
||||||
from pretix.base.services.mail import mail, prefix_subject
|
from pretix.base.services.mail import mail, prefix_subject
|
||||||
@@ -207,6 +207,7 @@ class OrganizerDetail(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixin
|
|||||||
'organizer').prefetch_related(
|
'organizer').prefetch_related(
|
||||||
'organizer', '_settings_objects', 'organizer___settings_objects',
|
'organizer', '_settings_objects', 'organizer___settings_objects',
|
||||||
'organizer__meta_properties',
|
'organizer__meta_properties',
|
||||||
|
'limit_sales_channels',
|
||||||
Prefetch(
|
Prefetch(
|
||||||
'meta_values',
|
'meta_values',
|
||||||
EventMetaValue.objects.select_related('property'),
|
EventMetaValue.objects.select_related('property'),
|
||||||
@@ -237,6 +238,7 @@ class OrganizerDetail(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixin
|
|||||||
self.filter_form['meta_{}'.format(p.name)] for p in
|
self.filter_form['meta_{}'.format(p.name)] for p in
|
||||||
self.organizer.meta_properties.filter(filter_allowed=True)
|
self.organizer.meta_properties.filter(filter_allowed=True)
|
||||||
]
|
]
|
||||||
|
ctx['sales_channels'] = self.request.organizer.sales_channels.all()
|
||||||
return ctx
|
return ctx
|
||||||
|
|
||||||
|
|
||||||
@@ -595,6 +597,13 @@ class OrganizerCreate(CreateView):
|
|||||||
})
|
})
|
||||||
|
|
||||||
|
|
||||||
|
def available_plugins(organizer):
|
||||||
|
from pretix.base.plugins import get_all_plugins
|
||||||
|
|
||||||
|
return (p for p in get_all_plugins(organizer=organizer) if not p.name.startswith('.')
|
||||||
|
and getattr(p, 'visible', True))
|
||||||
|
|
||||||
|
|
||||||
class OrganizerPlugins(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixin, TemplateView, SingleObjectMixin):
|
class OrganizerPlugins(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixin, TemplateView, SingleObjectMixin):
|
||||||
model = Organizer
|
model = Organizer
|
||||||
context_object_name = 'organizer'
|
context_object_name = 'organizer'
|
||||||
@@ -604,12 +613,6 @@ class OrganizerPlugins(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixi
|
|||||||
def get_object(self, queryset=None) -> Organizer:
|
def get_object(self, queryset=None) -> Organizer:
|
||||||
return self.request.organizer
|
return self.request.organizer
|
||||||
|
|
||||||
def available_plugins(self, organizer):
|
|
||||||
from pretix.base.plugins import get_all_plugins
|
|
||||||
|
|
||||||
return (p for p in get_all_plugins(organizer=organizer) if not p.name.startswith('.')
|
|
||||||
and getattr(p, 'visible', True))
|
|
||||||
|
|
||||||
def prepare_links(self, pluginmeta, key):
|
def prepare_links(self, pluginmeta, key):
|
||||||
links = getattr(pluginmeta, key, [])
|
links = getattr(pluginmeta, key, [])
|
||||||
try:
|
try:
|
||||||
@@ -635,7 +638,7 @@ class OrganizerPlugins(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixi
|
|||||||
from pretix.base.plugins import CATEGORY_LABELS, CATEGORY_ORDER
|
from pretix.base.plugins import CATEGORY_LABELS, CATEGORY_ORDER
|
||||||
|
|
||||||
context = super().get_context_data(*args, **kwargs)
|
context = super().get_context_data(*args, **kwargs)
|
||||||
plugins = list(self.available_plugins(self.object))
|
plugins = list(available_plugins(self.object))
|
||||||
|
|
||||||
active_counter = Counter()
|
active_counter = Counter()
|
||||||
events_total = 0
|
events_total = 0
|
||||||
@@ -683,7 +686,7 @@ class OrganizerPlugins(OrganizerDetailViewMixin, OrganizerPermissionRequiredMixi
|
|||||||
self.object = self.get_object()
|
self.object = self.get_object()
|
||||||
|
|
||||||
plugins_available = {
|
plugins_available = {
|
||||||
p.module: p for p in self.available_plugins(self.object)
|
p.module: p for p in available_plugins(self.object)
|
||||||
}
|
}
|
||||||
choose_events_next = False
|
choose_events_next = False
|
||||||
with transaction.atomic():
|
with transaction.atomic():
|
||||||
@@ -784,12 +787,6 @@ class OrganizerPluginEvents(OrganizerDetailViewMixin, OrganizerPermissionRequire
|
|||||||
}
|
}
|
||||||
return kwargs
|
return kwargs
|
||||||
|
|
||||||
def available_plugins(self, organizer):
|
|
||||||
from pretix.base.plugins import get_all_plugins
|
|
||||||
|
|
||||||
return (p for p in get_all_plugins(organizer=organizer) if not p.name.startswith('.')
|
|
||||||
and getattr(p, 'visible', True))
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
return super().get_context_data(
|
return super().get_context_data(
|
||||||
plugin=self.plugin,
|
plugin=self.plugin,
|
||||||
@@ -797,12 +794,10 @@ class OrganizerPluginEvents(OrganizerDetailViewMixin, OrganizerPermissionRequire
|
|||||||
)
|
)
|
||||||
|
|
||||||
def dispatch(self, request, *args, **kwargs):
|
def dispatch(self, request, *args, **kwargs):
|
||||||
plugins_available = {
|
try:
|
||||||
p.module: p for p in self.available_plugins(self.request.organizer)
|
self.plugin = next(p for p in available_plugins(self.request.organizer) if p.module == kwargs["plugin"])
|
||||||
}
|
except StopIteration:
|
||||||
if kwargs["plugin"] not in plugins_available:
|
|
||||||
raise Http404(_("Unknown plugin."))
|
raise Http404(_("Unknown plugin."))
|
||||||
self.plugin = plugins_available[kwargs["plugin"]]
|
|
||||||
level = getattr(self.plugin, "level", PLUGIN_LEVEL_EVENT)
|
level = getattr(self.plugin, "level", PLUGIN_LEVEL_EVENT)
|
||||||
if level == PLUGIN_LEVEL_ORGANIZER:
|
if level == PLUGIN_LEVEL_ORGANIZER:
|
||||||
raise Http404(_("This plugin can only be enabled for the entire organizer account."))
|
raise Http404(_("This plugin can only be enabled for the entire organizer account."))
|
||||||
@@ -833,6 +828,9 @@ class OrganizerPluginEvents(OrganizerDetailViewMixin, OrganizerPermissionRequire
|
|||||||
logentries_to_save = []
|
logentries_to_save = []
|
||||||
|
|
||||||
for e in self.request.organizer.events.filter(pk__in=events_to_enable):
|
for e in self.request.organizer.events.filter(pk__in=events_to_enable):
|
||||||
|
if not plugin_is_available(self.plugin, organizer=self.request.organizer, event=e):
|
||||||
|
messages.warning(self.request, _("This plugin cannot be activated for event {}.").format(e.name))
|
||||||
|
continue
|
||||||
logentries_to_save.append(
|
logentries_to_save.append(
|
||||||
e.log_action('pretix.event.plugins.enabled', user=self.request.user, data={'plugin': self.plugin.module}, save=False)
|
e.log_action('pretix.event.plugins.enabled', user=self.request.user, data={'plugin': self.plugin.module}, save=False)
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -292,6 +292,7 @@ class BaseEditorView(EventPermissionRequiredMixin, TemplateView):
|
|||||||
ctx['layout'] = json.dumps(self.get_current_layout())
|
ctx['layout'] = json.dumps(self.get_current_layout())
|
||||||
ctx['title'] = self.title
|
ctx['title'] = self.title
|
||||||
ctx['locales'] = [p for p in settings.LANGUAGES if p[0] in self.request.event.settings.locales]
|
ctx['locales'] = [p for p in settings.LANGUAGES if p[0] in self.request.event.settings.locales]
|
||||||
|
ctx['maxfilesize'] = self.maxfilesize
|
||||||
return ctx
|
return ctx
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -131,7 +131,7 @@ class VoucherList(PaginationMixin, EventPermissionRequiredMixin, ListView):
|
|||||||
elif v.quota:
|
elif v.quota:
|
||||||
prod = _('Any product in quota "{quota}"').format(quota=str(v.quota.name))
|
prod = _('Any product in quota "{quota}"').format(quota=str(v.quota.name))
|
||||||
else:
|
else:
|
||||||
prod = _('Any product')
|
prod = ""
|
||||||
row = [
|
row = [
|
||||||
v.code,
|
v.code,
|
||||||
v.valid_until.isoformat() if v.valid_until else "",
|
v.valid_until.isoformat() if v.valid_until else "",
|
||||||
|
|||||||
@@ -280,11 +280,12 @@ class WaitingListView(EventPermissionRequiredMixin, WaitingListQuerySetMixin, Pa
|
|||||||
block_quota=True,
|
block_quota=True,
|
||||||
item_id=wle.item_id,
|
item_id=wle.item_id,
|
||||||
subevent=wle.subevent_id,
|
subevent=wle.subevent_id,
|
||||||
waitinglistentries__isnull=False
|
waitinglistentries__isnull=False,
|
||||||
|
seat__isnull=True
|
||||||
).aggregate(free=Sum(F('max_usages') - F('redeemed')))['free'] or 0
|
).aggregate(free=Sum(F('max_usages') - F('redeemed')))['free'] or 0
|
||||||
free_seats = num_free_seats_for_product - num_valid_vouchers_for_product
|
free_seats = num_free_seats_for_product - num_valid_vouchers_for_product
|
||||||
wle.availability = (
|
wle.availability = (
|
||||||
Quota.AVAILABILITY_GONE if free_seats == 0 else wle.availability[0],
|
Quota.AVAILABILITY_GONE if free_seats < 1 else wle.availability[0],
|
||||||
min(free_seats, wle.availability[1]) if wle.availability[1] is not None else free_seats,
|
min(free_seats, wle.availability[1]) if wle.availability[1] is not None else free_seats,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -4,16 +4,16 @@ msgstr ""
|
|||||||
"Project-Id-Version: 1\n"
|
"Project-Id-Version: 1\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
||||||
"PO-Revision-Date: 2026-02-19 22:00+0000\n"
|
"PO-Revision-Date: 2026-03-05 20:00+0000\n"
|
||||||
"Last-Translator: Mie Frydensbjerg <mif@aarhus.dk>\n"
|
"Last-Translator: Mie Frydensbjerg <mif@aarhus.dk>\n"
|
||||||
"Language-Team: Danish <https://translate.pretix.eu/projects/pretix/pretix/da/"
|
"Language-Team: Danish <https://translate.pretix.eu/projects/pretix/pretix/"
|
||||||
">\n"
|
"da/>\n"
|
||||||
"Language: da\n"
|
"Language: da\n"
|
||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
||||||
"X-Generator: Weblate 5.16\n"
|
"X-Generator: Weblate 5.16.1\n"
|
||||||
|
|
||||||
#: pretix/_base_settings.py:87
|
#: pretix/_base_settings.py:87
|
||||||
msgid "English"
|
msgid "English"
|
||||||
@@ -34285,7 +34285,7 @@ msgstr ""
|
|||||||
#: pretix/presale/templates/pretixpresale/event/voucher.html:293
|
#: pretix/presale/templates/pretixpresale/event/voucher.html:293
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "minimum amount to order: %(num)s"
|
msgid "minimum amount to order: %(num)s"
|
||||||
msgstr ""
|
msgstr "Minimumsbestilling: %(num)s"
|
||||||
|
|
||||||
#: pretix/presale/templates/pretixpresale/event/fragment_addon_choice.html:76
|
#: pretix/presale/templates/pretixpresale/event/fragment_addon_choice.html:76
|
||||||
#: pretix/presale/templates/pretixpresale/event/fragment_addon_choice.html:160
|
#: pretix/presale/templates/pretixpresale/event/fragment_addon_choice.html:160
|
||||||
|
|||||||
@@ -5,8 +5,8 @@ msgstr ""
|
|||||||
"Project-Id-Version: 1\n"
|
"Project-Id-Version: 1\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
||||||
"PO-Revision-Date: 2026-02-24 12:07+0000\n"
|
"PO-Revision-Date: 2026-03-07 23:00+0000\n"
|
||||||
"Last-Translator: Raphael Michel <michel@rami.io>\n"
|
"Last-Translator: argonimos <jonas@pfeiffer-wagner.de>\n"
|
||||||
"Language-Team: German <https://translate.pretix.eu/projects/pretix/pretix/"
|
"Language-Team: German <https://translate.pretix.eu/projects/pretix/pretix/"
|
||||||
"de/>\n"
|
"de/>\n"
|
||||||
"Language: de\n"
|
"Language: de\n"
|
||||||
@@ -14,7 +14,7 @@ msgstr ""
|
|||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
||||||
"X-Generator: Weblate 5.16\n"
|
"X-Generator: Weblate 5.16.2\n"
|
||||||
"X-Poedit-Bookmarks: -1,-1,904,-1,-1,-1,-1,-1,-1,-1\n"
|
"X-Poedit-Bookmarks: -1,-1,904,-1,-1,-1,-1,-1,-1,-1\n"
|
||||||
|
|
||||||
#: pretix/_base_settings.py:87
|
#: pretix/_base_settings.py:87
|
||||||
@@ -3845,7 +3845,7 @@ msgstr "Restbetrag"
|
|||||||
#, python-brace-format
|
#, python-brace-format
|
||||||
msgctxt "invoice"
|
msgctxt "invoice"
|
||||||
msgid "Invoice period: {daterange}"
|
msgid "Invoice period: {daterange}"
|
||||||
msgstr "Rechungsperiode: {daterange}"
|
msgstr "Rechnungsperiode: {daterange}"
|
||||||
|
|
||||||
#: pretix/base/invoicing/pdf.py:1039
|
#: pretix/base/invoicing/pdf.py:1039
|
||||||
msgctxt "invoice"
|
msgctxt "invoice"
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
||||||
"PO-Revision-Date: 2026-03-02 10:00+0000\n"
|
"PO-Revision-Date: 2026-03-09 12:52+0000\n"
|
||||||
"Last-Translator: Hijiri Umemoto <hijiri@umemoto.org>\n"
|
"Last-Translator: Hijiri Umemoto <hijiri@umemoto.org>\n"
|
||||||
"Language-Team: Japanese <https://translate.pretix.eu/projects/pretix/pretix/"
|
"Language-Team: Japanese <https://translate.pretix.eu/projects/pretix/pretix/"
|
||||||
"ja/>\n"
|
"ja/>\n"
|
||||||
@@ -17,7 +17,7 @@ msgstr ""
|
|||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=1; plural=0;\n"
|
"Plural-Forms: nplurals=1; plural=0;\n"
|
||||||
"X-Generator: Weblate 5.16.1\n"
|
"X-Generator: Weblate 5.16.2\n"
|
||||||
|
|
||||||
#: pretix/_base_settings.py:87
|
#: pretix/_base_settings.py:87
|
||||||
msgid "English"
|
msgid "English"
|
||||||
@@ -11448,7 +11448,7 @@ msgstr ""
|
|||||||
"{event}のご注文が完了しました。無料製品のみのご注文のため、\n"
|
"{event}のご注文が完了しました。無料製品のみのご注文のため、\n"
|
||||||
"お支払いは不要です。\n"
|
"お支払いは不要です。\n"
|
||||||
"\n"
|
"\n"
|
||||||
"注文の詳細の変更やステータス確認は、以下のURLから行えます:\n"
|
"注文の詳細の変更やステータス確認は、以下のURLから行えます\n"
|
||||||
"{url}\n"
|
"{url}\n"
|
||||||
"\n"
|
"\n"
|
||||||
"よろしくお願いいたします。\n"
|
"よろしくお願いいたします。\n"
|
||||||
@@ -11750,7 +11750,7 @@ msgstr ""
|
|||||||
"{event}のご注文のお支払いを受け取りました。\n"
|
"{event}のご注文のお支払いを受け取りました。\n"
|
||||||
"\n"
|
"\n"
|
||||||
"残念ながら、受け取った金額は必要な全額よりも少ないです。\n"
|
"残念ながら、受け取った金額は必要な全額よりも少ないです。\n"
|
||||||
"したがって、追加の**{pending_sum}**の支払いが不足しているため、\n"
|
"したがって、追加の **{pending_sum}** の支払いが不足しているため、\n"
|
||||||
"ご注文は未払いと見なされます。\n"
|
"ご注文は未払いと見なされます。\n"
|
||||||
"\n"
|
"\n"
|
||||||
"お支払い情報やご注文の状況は、以下のURLでご確認いただけます。\n"
|
"お支払い情報やご注文の状況は、以下のURLでご確認いただけます。\n"
|
||||||
@@ -18428,7 +18428,7 @@ msgid ""
|
|||||||
"Do you really want to grant the application <strong>%(application)s</strong> "
|
"Do you really want to grant the application <strong>%(application)s</strong> "
|
||||||
"access to your pretix account?"
|
"access to your pretix account?"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"本当にアプリケーション<strong>%(application)s</strong>にPretixアカウントへの"
|
"本当にアプリケーション<strong>%(application)s</strong>にpretixアカウントへの"
|
||||||
"アクセスを許可しますか?"
|
"アクセスを許可しますか?"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/auth/oauth_authorization.html:24
|
#: pretix/control/templates/pretixcontrol/auth/oauth_authorization.html:24
|
||||||
@@ -24692,7 +24692,7 @@ msgstr "顧客履歴"
|
|||||||
#: pretix/control/templates/pretixcontrol/organizers/customer_anonymize.html:11
|
#: pretix/control/templates/pretixcontrol/organizers/customer_anonymize.html:11
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Anonymize customer #%(id)s"
|
msgid "Anonymize customer #%(id)s"
|
||||||
msgstr "顧客のID #%(id)s を匿名化"
|
msgstr "顧客 #%(id)s を匿名化"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/organizers/customer_anonymize.html:16
|
#: pretix/control/templates/pretixcontrol/organizers/customer_anonymize.html:16
|
||||||
msgid "Are you sure you want to anonymize this customer account?"
|
msgid "Are you sure you want to anonymize this customer account?"
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ msgstr ""
|
|||||||
"Project-Id-Version: PACKAGE VERSION\n"
|
"Project-Id-Version: PACKAGE VERSION\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
"POT-Creation-Date: 2026-02-24 11:50+0000\n"
|
||||||
"PO-Revision-Date: 2026-03-04 16:57+0000\n"
|
"PO-Revision-Date: 2026-03-09 12:52+0000\n"
|
||||||
"Last-Translator: Ruud Hendrickx <ruud@leckxicon.eu>\n"
|
"Last-Translator: Ruud Hendrickx <ruud@leckxicon.eu>\n"
|
||||||
"Language-Team: Dutch (Belgium) <https://translate.pretix.eu/projects/pretix/"
|
"Language-Team: Dutch (Belgium) <https://translate.pretix.eu/projects/pretix/"
|
||||||
"pretix/nl_BE/>\n"
|
"pretix/nl_BE/>\n"
|
||||||
@@ -17,7 +17,7 @@ msgstr ""
|
|||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
"Plural-Forms: nplurals=2; plural=n != 1;\n"
|
||||||
"X-Generator: Weblate 5.16.1\n"
|
"X-Generator: Weblate 5.16.2\n"
|
||||||
|
|
||||||
#: pretix/_base_settings.py:87
|
#: pretix/_base_settings.py:87
|
||||||
msgid "English"
|
msgid "English"
|
||||||
@@ -26981,6 +26981,10 @@ msgid ""
|
|||||||
"the affected data in your legislation, e.g. for reasons of taxation. In many "
|
"the affected data in your legislation, e.g. for reasons of taxation. In many "
|
||||||
"countries, you need to keep some data in the live system in case of an audit."
|
"countries, you need to keep some data in the live system in case of an audit."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
"Het is uw eigen verantwoordelijkheid om te controleren of u de gegevens "
|
||||||
|
"volgens uw wetgeving mag verwijderen, bijvoorbeeld om fiscale redenen. In "
|
||||||
|
"veel landen moet u bepaalde gegevens in het livesysteem bewaren voor het "
|
||||||
|
"geval er een audit plaatsvindt."
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/shredder/index.html:32
|
#: pretix/control/templates/pretixcontrol/shredder/index.html:32
|
||||||
msgid ""
|
msgid ""
|
||||||
@@ -26988,81 +26992,87 @@ msgid ""
|
|||||||
"to store it offline. Some kinds of data (such as some payment information) "
|
"to store it offline. Some kinds of data (such as some payment information) "
|
||||||
"as well as historical log data cannot be downloaded at the moment."
|
"as well as historical log data cannot be downloaded at the moment."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
"U kunt voor de meeste categorieën de gegevens gedeeltelijk downloaden om ze "
|
||||||
|
"offline op te slaan. Sommige soorten gegevens (bijvoorbeeld sommige "
|
||||||
|
"betalingsinformatie) en historische loggegevens kunnen momenteel niet worden "
|
||||||
|
"gedownload."
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/shredder/index.html:46
|
#: pretix/control/templates/pretixcontrol/shredder/index.html:46
|
||||||
msgid "Data selection"
|
msgid "Data selection"
|
||||||
msgstr ""
|
msgstr "Gegevensselectie"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/shredder/index.html:63
|
#: pretix/control/templates/pretixcontrol/shredder/index.html:63
|
||||||
msgid ""
|
msgid ""
|
||||||
"We recommend not to remove this data because you might need it in case of a "
|
"We recommend not to remove this data because you might need it in case of a "
|
||||||
"tax audit."
|
"tax audit."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
"We raden aan om deze gegevens niet te verwijderen, omdat u ze mogelijk nodig "
|
||||||
|
"hebt bij een belastingaudit."
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:10
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:10
|
||||||
msgctxt "subevent"
|
msgctxt "subevent"
|
||||||
msgid "Create multiple dates"
|
msgid "Create multiple dates"
|
||||||
msgstr ""
|
msgstr "Meerdere datums aanmaken"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:35
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:35
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:146
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:146
|
||||||
msgid "Repetition rule"
|
msgid "Repetition rule"
|
||||||
msgstr ""
|
msgstr "Regel voor herhaling"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:81
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:81
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:192
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:192
|
||||||
#, python-format
|
#, python-format
|
||||||
msgid "Repeat every %(interval)s %(freq)s, starting at %(start)s."
|
msgid "Repeat every %(interval)s %(freq)s, starting at %(start)s."
|
||||||
msgstr ""
|
msgstr "Herhaal ieder(e) %(interval)s %(freq)s, beginnend op %(start)s."
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:258
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:258
|
||||||
msgctxt "subevent"
|
msgctxt "subevent"
|
||||||
msgid "Preview"
|
msgid "Preview"
|
||||||
msgstr ""
|
msgstr "Voorbeeldweergave"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:265
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:265
|
||||||
msgctxt "subevent"
|
msgctxt "subevent"
|
||||||
msgid "Times"
|
msgid "Times"
|
||||||
msgstr ""
|
msgstr "Tijden"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:339
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:339
|
||||||
msgid "Start of first slot"
|
msgid "Start of first slot"
|
||||||
msgstr ""
|
msgstr "Begin van eerste tijdsslot"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:345
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:345
|
||||||
msgid "End of time slots"
|
msgid "End of time slots"
|
||||||
msgstr ""
|
msgstr "Einde van tijdsslots"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:351
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:351
|
||||||
msgid "Length of slots"
|
msgid "Length of slots"
|
||||||
msgstr ""
|
msgstr "Lengte van tijdsslots"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:360
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:360
|
||||||
msgid "Break between slots"
|
msgid "Break between slots"
|
||||||
msgstr ""
|
msgstr "Pauze tussen tijdsslots"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:370
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:370
|
||||||
msgid "Create"
|
msgid "Create"
|
||||||
msgstr ""
|
msgstr "Aanmaken"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:377
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:377
|
||||||
msgid "Add a single time slot"
|
msgid "Add a single time slot"
|
||||||
msgstr ""
|
msgstr "Eén tijdsslot toevoegen"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:379
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:379
|
||||||
msgid "Add many time slots"
|
msgid "Add many time slots"
|
||||||
msgstr ""
|
msgstr "Meerdere tijdsslots toevoegen"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:481
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:481
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk_edit.html:266
|
#: pretix/control/templates/pretixcontrol/subevents/bulk_edit.html:266
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/detail.html:124
|
#: pretix/control/templates/pretixcontrol/subevents/detail.html:124
|
||||||
msgid "Add a new quota"
|
msgid "Add a new quota"
|
||||||
msgstr ""
|
msgstr "Nieuw quotum toevoegen"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:485
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:485
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/detail.html:128
|
#: pretix/control/templates/pretixcontrol/subevents/detail.html:128
|
||||||
msgid "Product settings"
|
msgid "Product settings"
|
||||||
msgstr ""
|
msgstr "Productinstellingen"
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:487
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:487
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/detail.html:130
|
#: pretix/control/templates/pretixcontrol/subevents/detail.html:130
|
||||||
@@ -27070,6 +27080,8 @@ msgid ""
|
|||||||
"These settings are optional, if you leave them empty, the default values "
|
"These settings are optional, if you leave them empty, the default values "
|
||||||
"from the product settings will be used."
|
"from the product settings will be used."
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
"Deze instellingen zijn optioneel. Als u deze instellingen leeg laat, zullen "
|
||||||
|
"de standaardwaarden uit de productinstellingen worden gebruikt."
|
||||||
|
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:523
|
#: pretix/control/templates/pretixcontrol/subevents/bulk.html:523
|
||||||
#: pretix/control/templates/pretixcontrol/subevents/detail.html:166
|
#: pretix/control/templates/pretixcontrol/subevents/detail.html:166
|
||||||
|
|||||||
@@ -38,13 +38,10 @@ from i18nfield.strings import LazyI18nString
|
|||||||
|
|
||||||
from pretix.base.email import get_email_context
|
from pretix.base.email import get_email_context
|
||||||
from pretix.base.i18n import language
|
from pretix.base.i18n import language
|
||||||
from pretix.base.models import (
|
from pretix.base.models import Checkin, Event, InvoiceAddress, Order, User
|
||||||
CachedFile, Checkin, Event, InvoiceAddress, Order, User,
|
|
||||||
)
|
|
||||||
from pretix.base.services.mail import mail
|
from pretix.base.services.mail import mail
|
||||||
from pretix.base.services.tasks import ProfiledEventTask
|
from pretix.base.services.tasks import ProfiledEventTask
|
||||||
from pretix.celery_app import app
|
from pretix.celery_app import app
|
||||||
from pretix.helpers.format import format_map
|
|
||||||
|
|
||||||
|
|
||||||
def _chunks(lst, n):
|
def _chunks(lst, n):
|
||||||
@@ -64,7 +61,6 @@ def send_mails_to_orders(event: Event, user: int, subject: dict, message: dict,
|
|||||||
user = User.objects.get(pk=user) if user else None
|
user = User.objects.get(pk=user) if user else None
|
||||||
subject = LazyI18nString(subject)
|
subject = LazyI18nString(subject)
|
||||||
message = LazyI18nString(message)
|
message = LazyI18nString(message)
|
||||||
attachments_for_log = [cf.filename for cf in CachedFile.objects.filter(pk__in=attachments)] if attachments else []
|
|
||||||
|
|
||||||
def _send_to_order(o):
|
def _send_to_order(o):
|
||||||
send_to_order = recipients in ('both', 'orders')
|
send_to_order = recipients in ('both', 'orders')
|
||||||
@@ -122,7 +118,7 @@ def send_mails_to_orders(event: Event, user: int, subject: dict, message: dict,
|
|||||||
|
|
||||||
with language(o.locale, event.settings.region):
|
with language(o.locale, event.settings.region):
|
||||||
email_context = get_email_context(event=event, order=o, invoice_address=ia, position=p)
|
email_context = get_email_context(event=event, order=o, invoice_address=ia, position=p)
|
||||||
mail(
|
outgoing_mail = mail(
|
||||||
p.attendee_email,
|
p.attendee_email,
|
||||||
subject,
|
subject,
|
||||||
message,
|
message,
|
||||||
@@ -135,25 +131,17 @@ def send_mails_to_orders(event: Event, user: int, subject: dict, message: dict,
|
|||||||
attach_ical=attach_ical,
|
attach_ical=attach_ical,
|
||||||
attach_cached_files=attachments
|
attach_cached_files=attachments
|
||||||
)
|
)
|
||||||
o.log_action(
|
if outgoing_mail:
|
||||||
'pretix.plugins.sendmail.order.email.sent.attendee',
|
o.log_action(
|
||||||
user=user,
|
'pretix.plugins.sendmail.order.email.sent.attendee',
|
||||||
data={
|
user=user,
|
||||||
'position': p.positionid,
|
data=outgoing_mail.log_data(),
|
||||||
'subject': format_map(subject.localize(o.locale), email_context),
|
)
|
||||||
'message': format_map(message.localize(o.locale), email_context),
|
|
||||||
'recipient': p.attendee_email,
|
|
||||||
'attach_tickets': attach_tickets,
|
|
||||||
'attach_ical': attach_ical,
|
|
||||||
'attach_other_files': [],
|
|
||||||
'attach_cached_files': attachments_for_log,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
if send_to_order and o.email:
|
if send_to_order and o.email:
|
||||||
with language(o.locale, event.settings.region):
|
with language(o.locale, event.settings.region):
|
||||||
email_context = get_email_context(event=event, order=o, invoice_address=ia)
|
email_context = get_email_context(event=event, order=o, invoice_address=ia)
|
||||||
mail(
|
outgoing_mail = mail(
|
||||||
o.email,
|
o.email,
|
||||||
subject,
|
subject,
|
||||||
message,
|
message,
|
||||||
@@ -165,19 +153,12 @@ def send_mails_to_orders(event: Event, user: int, subject: dict, message: dict,
|
|||||||
attach_ical=attach_ical,
|
attach_ical=attach_ical,
|
||||||
attach_cached_files=attachments,
|
attach_cached_files=attachments,
|
||||||
)
|
)
|
||||||
o.log_action(
|
if outgoing_mail:
|
||||||
'pretix.plugins.sendmail.order.email.sent',
|
o.log_action(
|
||||||
user=user,
|
'pretix.plugins.sendmail.order.email.sent',
|
||||||
data={
|
user=user,
|
||||||
'subject': format_map(subject.localize(o.locale), email_context),
|
data=outgoing_mail.log_data(),
|
||||||
'message': format_map(message.localize(o.locale), email_context),
|
)
|
||||||
'recipient': o.email,
|
|
||||||
'attach_tickets': attach_tickets,
|
|
||||||
'attach_ical': attach_ical,
|
|
||||||
'attach_other_files': [],
|
|
||||||
'attach_cached_files': attachments_for_log,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
for chunk in _chunks(objects, 1000):
|
for chunk in _chunks(objects, 1000):
|
||||||
orders = Order.objects.filter(pk__in=chunk, event=event)
|
orders = Order.objects.filter(pk__in=chunk, event=event)
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
<div class="panel-group" id="questions_accordion">
|
<div class="panel-group" id="questions_accordion">
|
||||||
{% if invoice_address_asked or event.settings.invoice_name_required %}
|
{% if invoice_address_asked or event.settings.invoice_name_required %}
|
||||||
{% if invoice_address_asked and not request.GET.generate_invoice == "true" and not event.settings.invoice_reissue_after_modify %}
|
{% if invoice_address_asked and not request.GET.generate_invoice == "true" and not invoice_generation_selfservice %}
|
||||||
<div class="alert alert-info">
|
<div class="alert alert-info">
|
||||||
{% blocktrans trimmed %}
|
{% blocktrans trimmed %}
|
||||||
Modifying your invoice address will not automatically generate a new invoice.
|
Modifying your invoice address will not automatically generate a new invoice.
|
||||||
|
|||||||
@@ -20,7 +20,6 @@
|
|||||||
{% bootstrap_form_errors timemachine_form "all" %}
|
{% bootstrap_form_errors timemachine_form "all" %}
|
||||||
|
|
||||||
<p>{% trans "Test your shop as if it were a different date and time." %}</p>
|
<p>{% trans "Test your shop as if it were a different date and time." %}</p>
|
||||||
<p>{% trans "Please note that the changed time is not taken into account for aspects of the shop that affect quotas, such as the validity period of carts and vouchers." %}</p>
|
|
||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-6">
|
<div class="col-md-6">
|
||||||
@@ -45,4 +44,4 @@
|
|||||||
<div class="clear"></div>
|
<div class="clear"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -909,6 +909,21 @@ class OrderModify(EventViewMixin, OrderDetailMixin, OrderQuestionsViewMixin, Tem
|
|||||||
def get(self, request, *args, **kwargs):
|
def get(self, request, *args, **kwargs):
|
||||||
return super().get(request, *args, **kwargs)
|
return super().get(request, *args, **kwargs)
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
ctx = super().get_context_data(
|
||||||
|
**kwargs,
|
||||||
|
)
|
||||||
|
|
||||||
|
ctx['invoice_generation_selfservice'] = (
|
||||||
|
self.request.event.settings.invoice_reissue_after_modify or
|
||||||
|
(
|
||||||
|
can_generate_invoice(self.request.event, self.order, ignore_payments=True) and
|
||||||
|
not self.order.invoices.exists()
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
return ctx
|
||||||
|
|
||||||
def dispatch(self, request, *args, **kwargs):
|
def dispatch(self, request, *args, **kwargs):
|
||||||
self.request = request
|
self.request = request
|
||||||
self.kwargs = kwargs
|
self.kwargs = kwargs
|
||||||
|
|||||||
12
src/pretix/static/npm_dir/package-lock.json
generated
12
src/pretix/static/npm_dir/package-lock.json
generated
@@ -2776,9 +2776,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/minimatch": {
|
"node_modules/minimatch": {
|
||||||
"version": "3.0.4",
|
"version": "3.1.5",
|
||||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz",
|
||||||
"integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
|
"integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==",
|
||||||
"optional": true,
|
"optional": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"brace-expansion": "^1.1.7"
|
"brace-expansion": "^1.1.7"
|
||||||
@@ -5642,9 +5642,9 @@
|
|||||||
"optional": true
|
"optional": true
|
||||||
},
|
},
|
||||||
"minimatch": {
|
"minimatch": {
|
||||||
"version": "3.0.4",
|
"version": "3.1.5",
|
||||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz",
|
||||||
"integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==",
|
"integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==",
|
||||||
"optional": true,
|
"optional": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
"brace-expansion": "^1.1.7"
|
"brace-expansion": "^1.1.7"
|
||||||
|
|||||||
@@ -173,8 +173,8 @@ $(function () {
|
|||||||
if (!dependents.transmission_peppol_participant_id.val()) {
|
if (!dependents.transmission_peppol_participant_id.val()) {
|
||||||
const fill_peppol_id = function () {
|
const fill_peppol_id = function () {
|
||||||
const vatId = dependents.vat_id.val();
|
const vatId = dependents.vat_id.val();
|
||||||
if (vatId && vatId.startsWith("BE") && dependents.transmission_type.val() === "peppol" && autofill_peppol_id) {
|
if (vatId && vatId.startsWith("BE") && dependents.transmission_type.val() === "peppol") {
|
||||||
dependents.transmission_peppol_participant_id.val("0201:" + vatId.substring(2))
|
dependents.transmission_peppol_participant_id.val("0208:" + vatId.substring(2))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
dependents.vat_id.add(dependents.transmission_type).on("change", fill_peppol_id);
|
dependents.vat_id.add(dependents.transmission_type).on("change", fill_peppol_id);
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ from stripe import error
|
|||||||
from tests.plugins.stripe.test_checkout import apple_domain_create
|
from tests.plugins.stripe.test_checkout import apple_domain_create
|
||||||
from tests.plugins.stripe.test_provider import MockedCharge
|
from tests.plugins.stripe.test_provider import MockedCharge
|
||||||
|
|
||||||
from pretix.base.models import InvoiceAddress, Order, OrderPosition
|
from pretix.base.models import InvoiceAddress, Order, OrderPosition, Team
|
||||||
from pretix.base.models.orders import OrderFee, OrderPayment, OrderRefund
|
from pretix.base.models.orders import OrderFee, OrderPayment, OrderRefund
|
||||||
|
|
||||||
|
|
||||||
@@ -180,6 +180,41 @@ def order2(event2, item2):
|
|||||||
return o
|
return o
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
@scopes_disabled()
|
||||||
|
def team2(organizer, event2):
|
||||||
|
team2 = Team.objects.create(
|
||||||
|
organizer=organizer,
|
||||||
|
name="Test-Team 2",
|
||||||
|
can_change_teams=True,
|
||||||
|
can_manage_gift_cards=True,
|
||||||
|
can_change_items=True,
|
||||||
|
can_create_events=True,
|
||||||
|
can_change_event_settings=True,
|
||||||
|
can_change_vouchers=True,
|
||||||
|
can_view_vouchers=True,
|
||||||
|
can_change_orders=True,
|
||||||
|
can_manage_customers=True,
|
||||||
|
can_manage_reusable_media=True,
|
||||||
|
can_change_organizer_settings=True,
|
||||||
|
|
||||||
|
)
|
||||||
|
team2.limit_events.add(event2)
|
||||||
|
team2.save()
|
||||||
|
return team2
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
@scopes_disabled()
|
||||||
|
def limited_token_client(client, team2):
|
||||||
|
team2.can_view_orders = True
|
||||||
|
team2.can_view_vouchers = True
|
||||||
|
team2.save()
|
||||||
|
t = team2.tokens.create(name='Foo')
|
||||||
|
client.credentials(HTTP_AUTHORIZATION='Token ' + t.token)
|
||||||
|
return client
|
||||||
|
|
||||||
|
|
||||||
TEST_ORDERPOSITION_RES = {
|
TEST_ORDERPOSITION_RES = {
|
||||||
"id": 1,
|
"id": 1,
|
||||||
"order": "FOO",
|
"order": "FOO",
|
||||||
@@ -987,8 +1022,64 @@ def test_refund_cancel(token_client, organizer, event, order):
|
|||||||
assert resp.status_code == 400
|
assert resp.status_code == 400
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
"endpoint_template, response_code",
|
||||||
|
[('/api/v1/organizers/{}/events/{}/orderpositions/', 403), ('/api/v1/organizers/{}/orderpositions/', 200)]
|
||||||
|
)
|
||||||
@pytest.mark.django_db
|
@pytest.mark.django_db
|
||||||
def test_orderposition_list(token_client, organizer, device, event, order, item, subevent, subevent2, question, django_assert_num_queries):
|
def test_orderposition_list_limited_read(
|
||||||
|
endpoint_template, response_code, limited_token_client, organizer, device, event, order, item, subevent, subevent2, question
|
||||||
|
):
|
||||||
|
endpoint = endpoint_template.format(organizer.slug, event.slug)
|
||||||
|
|
||||||
|
i2 = copy.copy(item)
|
||||||
|
i2.pk = None
|
||||||
|
i2.save()
|
||||||
|
with scopes_disabled():
|
||||||
|
var = item.variations.create(value="Children")
|
||||||
|
res = copy.copy(TEST_ORDERPOSITION_RES)
|
||||||
|
op = order.positions.first()
|
||||||
|
op.variation = var
|
||||||
|
op.save()
|
||||||
|
res["id"] = op.pk
|
||||||
|
res["item"] = item.pk
|
||||||
|
res["variation"] = var.pk
|
||||||
|
res["answers"][0]["question"] = question.pk
|
||||||
|
res["print_logs"][0]["id"] = op.print_logs.first().pk
|
||||||
|
res["print_logs"][0]["device_id"] = device.device_id
|
||||||
|
|
||||||
|
resp = limited_token_client.get(endpoint)
|
||||||
|
assert resp.status_code == response_code
|
||||||
|
if response_code == 200:
|
||||||
|
assert resp.json() == {'count': 0, 'next': None, 'previous': None, 'results': []}
|
||||||
|
else:
|
||||||
|
assert resp.json() == {'detail': 'You do not have permission to perform this action.'}
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize(
|
||||||
|
("endpoint_template", "endpoint_type"),
|
||||||
|
[
|
||||||
|
('/api/v1/organizers/{}/events/{}/orderpositions/', "event"),
|
||||||
|
('/api/v1/organizers/{}/orderpositions/', "organizer")
|
||||||
|
],
|
||||||
|
)
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_orderposition_list(
|
||||||
|
endpoint_template,
|
||||||
|
endpoint_type,
|
||||||
|
token_client,
|
||||||
|
organizer,
|
||||||
|
device,
|
||||||
|
event,
|
||||||
|
order,
|
||||||
|
item,
|
||||||
|
subevent,
|
||||||
|
subevent2,
|
||||||
|
question,
|
||||||
|
django_assert_num_queries
|
||||||
|
):
|
||||||
|
endpoint = endpoint_template.format(organizer.slug, event.slug)
|
||||||
|
|
||||||
i2 = copy.copy(item)
|
i2 = copy.copy(item)
|
||||||
i2.pk = None
|
i2.pk = None
|
||||||
i2.save()
|
i2.save()
|
||||||
@@ -1005,88 +1096,64 @@ def test_orderposition_list(token_client, organizer, device, event, order, item,
|
|||||||
res["answers"][0]["question"] = question.pk
|
res["answers"][0]["question"] = question.pk
|
||||||
res["print_logs"][0]["id"] = op.print_logs.first().pk
|
res["print_logs"][0]["id"] = op.print_logs.first().pk
|
||||||
res["print_logs"][0]["device_id"] = device.device_id
|
res["print_logs"][0]["device_id"] = device.device_id
|
||||||
|
if endpoint_type == "organizer":
|
||||||
|
res["event"] = event.slug
|
||||||
|
|
||||||
resp = token_client.get('/api/v1/organizers/{}/events/{}/orderpositions/'.format(organizer.slug, event.slug))
|
resp = token_client.get(endpoint)
|
||||||
assert resp.status_code == 200
|
assert resp.status_code == 200
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?order__status=n')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?order__status=n'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?order__status=p')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?order__status=p'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?item={}'.format(item.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?item={}'.format(organizer.slug, event.slug, item.pk))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?item__in={},{}'.format(item.pk, i2.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?item__in={},{}'.format(
|
|
||||||
organizer.slug, event.slug, item.pk, i2.pk
|
|
||||||
))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?item={}'.format(i2.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?item={}'.format(organizer.slug, event.slug, i2.pk))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?variation={}'.format(var.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?variation={}'.format(organizer.slug, event.slug, var.pk))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?variation={}'.format(var2.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?variation={}'.format(organizer.slug, event.slug, var2.pk))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?attendee_name=Peter')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?attendee_name=Peter'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?attendee_name=peter')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?attendee_name=peter'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?attendee_name=Mark')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?attendee_name=Mark'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?secret=z3fsn8jyufm5kpk768q69gkbyr5f4h6w')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?secret=z3fsn8jyufm5kpk768q69gkbyr5f4h6w'.format(
|
|
||||||
organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?secret=abc123')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?secret=abc123'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?pseudonymization_id=ABCDEFGHKL')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?pseudonymization_id=ABCDEFGHKL'.format(
|
|
||||||
organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?pseudonymization_id=FOO')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?pseudonymization_id=FOO'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?search=FO')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?search=FO'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?search=z3fsn8j')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?search=z3fsn8j'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?search=Peter')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?search=Peter'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?search=5f4h6w')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?search=5f4h6w'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?order=FOO')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?order=FOO'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?order=BAR')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?order=BAR'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?has_checkin=false')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?has_checkin=false'.format(organizer.slug, event.slug))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?has_checkin=true')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?has_checkin=true'.format(organizer.slug, event.slug))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
with scopes_disabled():
|
with scopes_disabled():
|
||||||
@@ -1103,33 +1170,28 @@ def test_orderposition_list(token_client, organizer, device, event, order, item,
|
|||||||
'gate': None,
|
'gate': None,
|
||||||
'type': 'entry'
|
'type': 'entry'
|
||||||
}]
|
}]
|
||||||
with django_assert_num_queries(16):
|
if '/events/' in endpoint:
|
||||||
resp = token_client.get(
|
with django_assert_num_queries(18):
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?has_checkin=true'.format(organizer.slug, event.slug)
|
resp = token_client.get(endpoint + '?has_checkin=true')
|
||||||
)
|
else:
|
||||||
|
with django_assert_num_queries(17):
|
||||||
|
resp = token_client.get(endpoint + '?has_checkin=true')
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
|
|
||||||
op.subevent = subevent
|
op.subevent = subevent
|
||||||
op.save()
|
op.save()
|
||||||
res['subevent'] = subevent.pk
|
res['subevent'] = subevent.pk
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?subevent={}'.format(subevent.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?subevent={}'.format(organizer.slug, event.slug, subevent.pk))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?subevent__in={},{}'.format(subevent.pk, subevent2.pk))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?subevent__in={},{}'.format(organizer.slug, event.slug,
|
|
||||||
subevent.pk, subevent2.pk))
|
|
||||||
assert [res] == resp.data['results']
|
assert [res] == resp.data['results']
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?subevent={}'.format(subevent.pk + 1))
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?subevent={}'.format(organizer.slug, event.slug,
|
|
||||||
subevent.pk + 1))
|
|
||||||
assert [] == resp.data['results']
|
assert [] == resp.data['results']
|
||||||
|
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?include_canceled_positions=false')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?include_canceled_positions=false'.format(organizer.slug, event.slug))
|
|
||||||
assert len(resp.data['results']) == 1
|
assert len(resp.data['results']) == 1
|
||||||
resp = token_client.get(
|
resp = token_client.get(endpoint + '?include_canceled_positions=true')
|
||||||
'/api/v1/organizers/{}/events/{}/orderpositions/?include_canceled_positions=true'.format(organizer.slug, event.slug))
|
|
||||||
assert len(resp.data['results']) == 2
|
assert len(resp.data['results']) == 2
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -29,6 +29,8 @@ from pretix.base.models import (
|
|||||||
Event, Item, ItemVariation, Organizer, Quota, Team, User, Voucher,
|
Event, Item, ItemVariation, Organizer, Quota, Team, User, Voucher,
|
||||||
WaitingListEntry,
|
WaitingListEntry,
|
||||||
)
|
)
|
||||||
|
from pretix.base.models.seating import Seat, SeatingPlan
|
||||||
|
from pretix.base.models.waitinglist import WaitingListException
|
||||||
from pretix.control.views.dashboards import waitinglist_widgets
|
from pretix.control.views.dashboards import waitinglist_widgets
|
||||||
|
|
||||||
|
|
||||||
@@ -55,11 +57,11 @@ def env():
|
|||||||
WaitingListEntry.objects.create(
|
WaitingListEntry.objects.create(
|
||||||
event=event, item=item1, email='success@example.org', voucher=v
|
event=event, item=item1, email='success@example.org', voucher=v
|
||||||
)
|
)
|
||||||
v = Voucher.objects.create(item=item1, event=event, block_quota=True, redeemed=0, valid_until=now() - timedelta(days=5))
|
v = Voucher.objects.create(item=item2, event=event, block_quota=True, redeemed=0, valid_until=now() - timedelta(days=5))
|
||||||
WaitingListEntry.objects.create(
|
WaitingListEntry.objects.create(
|
||||||
event=event, item=item2, email='expired@example.org', voucher=v
|
event=event, item=item2, email='expired@example.org', voucher=v
|
||||||
)
|
)
|
||||||
v = Voucher.objects.create(item=item1, event=event, block_quota=True, redeemed=0, valid_until=now() + timedelta(days=5))
|
v = Voucher.objects.create(item=item2, event=event, block_quota=True, redeemed=0, valid_until=now() + timedelta(days=5))
|
||||||
WaitingListEntry.objects.create(
|
WaitingListEntry.objects.create(
|
||||||
event=event, item=item2, email='valid@example.org', voucher=v
|
event=event, item=item2, email='valid@example.org', voucher=v
|
||||||
)
|
)
|
||||||
@@ -345,5 +347,75 @@ def test_dashboard(client, env):
|
|||||||
quota.items.add(env['item1'])
|
quota.items.add(env['item1'])
|
||||||
w = waitinglist_widgets(env['event'])
|
w = waitinglist_widgets(env['event'])
|
||||||
|
|
||||||
assert '1' in w[0]['content']
|
assert '2' in w[0]['content']
|
||||||
assert '5' in w[1]['content']
|
assert '5' in w[1]['content']
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_waitinglist_seat_calc(client, env):
|
||||||
|
item = env['item1']
|
||||||
|
event = env['event']
|
||||||
|
wle = env['wle']
|
||||||
|
|
||||||
|
SeatingPlan.objects.create(
|
||||||
|
name="Plan", organizer=event.organizer, layout="{}"
|
||||||
|
)
|
||||||
|
event.seat_category_mappings.create(
|
||||||
|
layout_category='Stalls', product=item
|
||||||
|
)
|
||||||
|
for i in range(2):
|
||||||
|
event.seats.create(seat_number=f"A{i}", product=item, seat_guid=f"A{i}")
|
||||||
|
|
||||||
|
quota = Quota.objects.create(event=event, size=10)
|
||||||
|
quota.items.add(item)
|
||||||
|
|
||||||
|
client.login(email='dummy@dummy.dummy', password='dummy')
|
||||||
|
|
||||||
|
# Calculated availability should not be more than number of available seats
|
||||||
|
response = client.get('/control/event/dummy/dummy/waitinglist/')
|
||||||
|
assert len(response.context['entries']) == 5
|
||||||
|
for entry in response.context['entries']:
|
||||||
|
assert entry.availability == (Quota.AVAILABILITY_OK, 2)
|
||||||
|
|
||||||
|
# Sending out a voucher reduces availability by 1
|
||||||
|
with scopes_disabled():
|
||||||
|
wle.send_voucher()
|
||||||
|
|
||||||
|
voucher = wle.voucher
|
||||||
|
assert voucher
|
||||||
|
|
||||||
|
response = client.get('/control/event/dummy/dummy/waitinglist/')
|
||||||
|
assert len(response.context['entries']) == 4
|
||||||
|
for entry in response.context['entries']:
|
||||||
|
assert entry.availability == (Quota.AVAILABILITY_OK, 1)
|
||||||
|
|
||||||
|
# Assigning a seat to a voucher does not decrease availability further
|
||||||
|
with scopes_disabled():
|
||||||
|
voucher.seat = Seat.objects.get(seat_guid="A0")
|
||||||
|
voucher.save()
|
||||||
|
|
||||||
|
response = client.get('/control/event/dummy/dummy/waitinglist/')
|
||||||
|
assert len(response.context['entries']) == 4
|
||||||
|
for entry in response.context['entries']:
|
||||||
|
assert entry.availability == (Quota.AVAILABILITY_OK, 1)
|
||||||
|
|
||||||
|
with scopes_disabled():
|
||||||
|
wle2 = WaitingListEntry.objects.filter(item=item, voucher__isnull=True).first()
|
||||||
|
wle2.send_voucher()
|
||||||
|
|
||||||
|
# Overbooking is handled correctly
|
||||||
|
# Regression test for calculation that used `not free_seats` instead of `free_seats < 1`
|
||||||
|
with scopes_disabled():
|
||||||
|
# Block seat
|
||||||
|
seat = Seat.objects.get(seat_guid="A1")
|
||||||
|
seat.blocked = True
|
||||||
|
seat.save()
|
||||||
|
|
||||||
|
response = client.get('/control/event/dummy/dummy/waitinglist/')
|
||||||
|
assert len(response.context['entries']) == 3
|
||||||
|
for entry in response.context['entries']:
|
||||||
|
assert entry.availability == (Quota.AVAILABILITY_GONE, -1)
|
||||||
|
|
||||||
|
with scopes_disabled(), pytest.raises(WaitingListException):
|
||||||
|
wle3 = WaitingListEntry.objects.filter(item=item, voucher__isnull=True).first()
|
||||||
|
wle3.send_voucher()
|
||||||
|
|||||||
Reference in New Issue
Block a user