mirror of
https://github.com/pretix/pretix.git
synced 2026-05-06 15:24:02 +00:00
158 lines
7.0 KiB
Python
158 lines
7.0 KiB
Python
from datetime import timedelta
|
|
|
|
from django.utils.crypto import get_random_string
|
|
from django.utils.timezone import now
|
|
from django.utils.translation import gettext_lazy
|
|
from rest_framework import serializers
|
|
from rest_framework.exceptions import ValidationError
|
|
|
|
from pretix.api.serializers.i18n import I18nAwareModelSerializer
|
|
from pretix.api.serializers.order import (
|
|
AnswerCreateSerializer, AnswerSerializer, InlineSeatSerializer,
|
|
)
|
|
from pretix.base.models import Quota, Seat
|
|
from pretix.base.models.orders import CartPosition
|
|
|
|
|
|
class CartPositionSerializer(I18nAwareModelSerializer):
|
|
answers = AnswerSerializer(many=True)
|
|
seat = InlineSeatSerializer()
|
|
|
|
class Meta:
|
|
model = CartPosition
|
|
fields = ('id', 'cart_id', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts',
|
|
'attendee_email', 'voucher', 'addon_to', 'subevent', 'datetime', 'expires', 'includes_tax',
|
|
'answers', 'seat')
|
|
|
|
|
|
class CartPositionCreateSerializer(I18nAwareModelSerializer):
|
|
answers = AnswerCreateSerializer(many=True, required=False)
|
|
expires = serializers.DateTimeField(required=False)
|
|
attendee_name = serializers.CharField(required=False, allow_null=True)
|
|
seat = serializers.CharField(required=False, allow_null=True)
|
|
sales_channel = serializers.CharField(required=False, default='sales_channel')
|
|
|
|
class Meta:
|
|
model = CartPosition
|
|
fields = ('cart_id', 'item', 'variation', 'price', 'attendee_name', 'attendee_name_parts', 'attendee_email',
|
|
'subevent', 'expires', 'includes_tax', 'answers', 'seat', 'sales_channel')
|
|
|
|
def create(self, validated_data):
|
|
answers_data = validated_data.pop('answers')
|
|
if not validated_data.get('cart_id'):
|
|
cid = "{}@api".format(get_random_string(48))
|
|
while CartPosition.objects.filter(cart_id=cid).exists():
|
|
cid = "{}@api".format(get_random_string(48))
|
|
validated_data['cart_id'] = cid
|
|
|
|
if not validated_data.get('expires'):
|
|
validated_data['expires'] = now() + timedelta(
|
|
minutes=self.context['event'].settings.get('reservation_time', as_type=int)
|
|
)
|
|
|
|
with self.context['event'].lock():
|
|
new_quotas = (validated_data.get('variation').quotas.filter(subevent=validated_data.get('subevent'))
|
|
if validated_data.get('variation')
|
|
else validated_data.get('item').quotas.filter(subevent=validated_data.get('subevent')))
|
|
if len(new_quotas) == 0:
|
|
raise ValidationError(
|
|
gettext_lazy('The product "{}" is not assigned to a quota.').format(
|
|
str(validated_data.get('item'))
|
|
)
|
|
)
|
|
for quota in new_quotas:
|
|
avail = quota.availability()
|
|
if avail[0] != Quota.AVAILABILITY_OK or (avail[1] is not None and avail[1] < 1):
|
|
raise ValidationError(
|
|
gettext_lazy('There is not enough quota available on quota "{}" to perform '
|
|
'the operation.').format(
|
|
quota.name
|
|
)
|
|
)
|
|
attendee_name = validated_data.pop('attendee_name', '')
|
|
if attendee_name and not validated_data.get('attendee_name_parts'):
|
|
validated_data['attendee_name_parts'] = {
|
|
'_legacy': attendee_name
|
|
}
|
|
|
|
seated = validated_data.get('item').seat_category_mappings.filter(subevent=validated_data.get('subevent')).exists()
|
|
if validated_data.get('seat'):
|
|
if not seated:
|
|
raise ValidationError('The specified product does not allow to choose a seat.')
|
|
try:
|
|
seat = self.context['event'].seats.get(seat_guid=validated_data['seat'], subevent=validated_data.get('subevent'))
|
|
except Seat.DoesNotExist:
|
|
raise ValidationError('The specified seat does not exist.')
|
|
except Seat.MultipleObjectsReturned:
|
|
raise ValidationError('The specified seat ID is not unique.')
|
|
else:
|
|
validated_data['seat'] = seat
|
|
if not seat.is_available(
|
|
sales_channel=validated_data.get('sales_channel', 'web'),
|
|
distance_ignore_cart_id=validated_data['cart_id'],
|
|
):
|
|
raise ValidationError(gettext_lazy('The selected seat "{seat}" is not available.').format(seat=seat.name))
|
|
elif seated:
|
|
raise ValidationError('The specified product requires to choose a seat.')
|
|
|
|
validated_data.pop('sales_channel')
|
|
cp = CartPosition.objects.create(event=self.context['event'], **validated_data)
|
|
|
|
for answ_data in answers_data:
|
|
options = answ_data.pop('options')
|
|
answ = cp.answers.create(**answ_data)
|
|
answ.options.add(*options)
|
|
return cp
|
|
|
|
def validate_cart_id(self, cid):
|
|
if cid and not cid.endswith('@api'):
|
|
raise ValidationError('Cart ID should end in @api or be empty.')
|
|
return cid
|
|
|
|
def validate_item(self, item):
|
|
if item.event != self.context['event']:
|
|
raise ValidationError(
|
|
'The specified item does not belong to this event.'
|
|
)
|
|
if not item.active:
|
|
raise ValidationError(
|
|
'The specified item is not active.'
|
|
)
|
|
return item
|
|
|
|
def validate_subevent(self, subevent):
|
|
if self.context['event'].has_subevents:
|
|
if not subevent:
|
|
raise ValidationError(
|
|
'You need to set a subevent.'
|
|
)
|
|
if subevent.event != self.context['event']:
|
|
raise ValidationError(
|
|
'The specified subevent does not belong to this event.'
|
|
)
|
|
elif subevent:
|
|
raise ValidationError(
|
|
'You cannot set a subevent for this event.'
|
|
)
|
|
return subevent
|
|
|
|
def validate(self, data):
|
|
if data.get('item'):
|
|
if data.get('item').has_variations:
|
|
if not data.get('variation'):
|
|
raise ValidationError('You should specify a variation for this item.')
|
|
else:
|
|
if data.get('variation').item != data.get('item'):
|
|
raise ValidationError(
|
|
'The specified variation does not belong to the specified item.'
|
|
)
|
|
elif data.get('variation'):
|
|
raise ValidationError(
|
|
'You cannot specify a variation for this item.'
|
|
)
|
|
if data.get('attendee_name') and data.get('attendee_name_parts'):
|
|
raise ValidationError(
|
|
{'attendee_name': ['Do not specify attendee_name if you specified attendee_name_parts.']}
|
|
)
|
|
return data
|