mirror of
https://github.com/pretix/pretix.git
synced 2026-06-10 01:15:05 +00:00
* change linked orderpositions to many-to-many * Update media views to list ops * return last op as fallback for linked_orderposition * add multi-op to export * update media-API * fix media-view filter * update control media forms * fix API orders * fix API orders matching media * remove cached_property linked_orderposition - keep only in API * fix media-issue signal * adapt checkin API for multiple orderpositions * remove unneeded comment * fix create/update logging * fix tests * fix more tests * fix code style * add label to reusablemedium * fix migration NOT NULL * fix tests * update docs * clarify docs updating multiple linked_orderpositions * clarify docs * no need to prefetch linked_orderpositions * improve readability * select_related order instead prefetch * add filter based on op.valid_from/until * rename secret to claim_token * Update docs for claim_token * unifiy deprecated style * Update reusablemedia.rst * Update reusablemedia.rst * Update reusablemedia.rst * fix missing claim_token in serializer * fix flake8 * add add_to_reusable_medium to order-serializer * fix tests regarding claim_token * fix flake8 * Clarify docs * list ops comma-separated in export * Add test for order-API add_to_reusable_medium * fix linked_orderpositions filter in checkinrpc * add test * Add help-text * fix multi-op media filter * fix flake8 * improve check * Fix sorting of reusable media type in overview * Add copy and qr button to reusable medium detail view * Rebase against origin/master * Add logentrytype reusable_medium.linked_orderposition.removed * add missing label_from_instance for SafeOrderPositionMultipleChoiceField * add tests for create with linked_orderposition * API add test for fallback-values in medium patch * fix flake8 * Fix indentation * fix migrations numbering * fix test * unify qutation marks * fix flake8 * micro-improve linked_op-removal-logging * simplify filter instead of annotate/get * Do not translate API-errors Co-authored-by: Raphael Michel <michel@pretix.eu> * Fix typos in doc Co-authored-by: Raphael Michel <mail@raphaelmichel.de> * Update versionchanged in docs Co-authored-by: Raphael Michel <michel@pretix.eu> * Change log to always added not changed * Add test for checkinrpc for ops out of timerang or canceled * improve tests mixing ops from different organizers * Fix logging of changed order_positions * properly log added/removed when using UI * refactor logging code * unify logging adding/removing ops via API * fix flake8 * remove unnecessary prefetch as already prefetched * optimize fetching ops * combine addon match and time-based validity match * fix combined valid and product check * re-number migrations * Apply suggestion from @raphaelm Co-authored-by: Raphael Michel <michel@pretix.eu> * fix flake8 * New attempt at logic * Improve op_candidate-selection for error message if no op matches check-in * Fix typo * fix valid_from start time being included * use the datetime parameter for the comparison time so that the simulator works too --------- Co-authored-by: Maximilian Richt <richt@pretix.eu> Co-authored-by: Martin Gross <gross@rami.io> Co-authored-by: Raphael Michel <michel@pretix.eu> Co-authored-by: Raphael Michel <mail@raphaelmichel.de>
162 lines
5.3 KiB
Python
162 lines
5.3 KiB
Python
#
|
|
# This file is part of pretix (Community Edition).
|
|
#
|
|
# Copyright (C) 2014-2020 Raphael Michel and contributors
|
|
# Copyright (C) 2020-today pretix GmbH and contributors
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
|
|
# Public License as published by the Free Software Foundation in version 3 of the License.
|
|
#
|
|
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
|
|
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
|
|
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
|
|
# this file, see <https://pretix.eu/about/en/license>.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
|
|
# <https://www.gnu.org/licenses/>.
|
|
#
|
|
from django.db import models
|
|
from django.db.models import Q
|
|
from django.utils.functional import cached_property
|
|
from django.utils.timezone import now
|
|
from django.utils.translation import gettext_lazy as _, pgettext_lazy
|
|
from django_scopes import ScopedManager
|
|
|
|
from pretix.base.media import MEDIA_TYPES
|
|
from pretix.base.models import LoggedModel
|
|
from pretix.base.models.customers import Customer
|
|
from pretix.base.models.giftcards import GiftCard
|
|
from pretix.base.models.orders import OrderPosition
|
|
from pretix.base.models.organizer import Organizer
|
|
|
|
|
|
class ReusableMediumQuerySet(models.QuerySet):
|
|
|
|
def active(self):
|
|
return self.filter(
|
|
Q(expires__isnull=True) | Q(expires__gte=now()),
|
|
active=True,
|
|
)
|
|
|
|
|
|
class ReusableMediumQuerySetManager(ScopedManager(organizer='organizer').__class__):
|
|
def __init__(self):
|
|
super().__init__()
|
|
self._queryset_class = ReusableMediumQuerySet
|
|
|
|
def active(self):
|
|
return self.get_queryset().active()
|
|
|
|
|
|
class ReusableMedium(LoggedModel):
|
|
id = models.BigAutoField(primary_key=True)
|
|
created = models.DateTimeField(auto_now_add=True)
|
|
updated = models.DateTimeField(auto_now=True)
|
|
|
|
organizer = models.ForeignKey(
|
|
Organizer,
|
|
related_name='reusable_media',
|
|
on_delete=models.PROTECT
|
|
)
|
|
|
|
type = models.CharField(
|
|
verbose_name=pgettext_lazy('reusable_medium', 'Media type'),
|
|
choices=((k, v) for k, v in MEDIA_TYPES.items()),
|
|
max_length=100,
|
|
)
|
|
identifier = models.CharField(
|
|
max_length=200,
|
|
verbose_name=pgettext_lazy('reusable_medium', 'Identifier'),
|
|
)
|
|
claim_token = models.CharField(
|
|
max_length=200,
|
|
verbose_name=pgettext_lazy('reusable_medium', 'Claim token'),
|
|
null=True, blank=True
|
|
)
|
|
label = models.CharField(
|
|
max_length=200,
|
|
verbose_name=pgettext_lazy('reusable_medium', 'Label'),
|
|
null=True, blank=True
|
|
)
|
|
|
|
active = models.BooleanField(
|
|
verbose_name=_('Active'),
|
|
default=True
|
|
)
|
|
expires = models.DateTimeField(
|
|
verbose_name=_('Expiration date'),
|
|
null=True, blank=True
|
|
)
|
|
|
|
customer = models.ForeignKey(
|
|
Customer,
|
|
null=True, blank=True,
|
|
related_name='reusable_media',
|
|
on_delete=models.SET_NULL,
|
|
verbose_name=_('Customer account'),
|
|
)
|
|
linked_orderpositions = models.ManyToManyField(
|
|
OrderPosition,
|
|
related_name='linked_media',
|
|
verbose_name=_('Linked tickets'),
|
|
help_text=_(
|
|
'If you link to more than one ticket, make sure there is no overlap in validity. '
|
|
'If multiple tickets are valid at once, this will lead to failed check-ins.'
|
|
)
|
|
)
|
|
linked_giftcard = models.ForeignKey(
|
|
GiftCard,
|
|
null=True, blank=True,
|
|
related_name='linked_media',
|
|
on_delete=models.SET_NULL,
|
|
verbose_name=_('Linked gift card'),
|
|
)
|
|
|
|
info = models.JSONField(
|
|
default=dict
|
|
)
|
|
notes = models.TextField(verbose_name=_('Notes'), null=True, blank=True)
|
|
|
|
objects = ReusableMediumQuerySetManager()
|
|
|
|
@cached_property
|
|
def media_type(self):
|
|
return MEDIA_TYPES[self.type]
|
|
|
|
@property
|
|
def is_expired(self):
|
|
return self.expires and self.expires > now()
|
|
|
|
class Meta:
|
|
unique_together = (("identifier", "type", "organizer"),)
|
|
indexes = [
|
|
models.Index(fields=("updated", "id"), name="pretixbase__updated_093277_idx"),
|
|
]
|
|
ordering = "identifier", "type", "organizer"
|
|
|
|
|
|
class MediumKeySet(models.Model):
|
|
organizer = models.ForeignKey('Organizer', on_delete=models.CASCADE, related_name='medium_key_sets')
|
|
public_id = models.BigIntegerField(
|
|
unique=True,
|
|
)
|
|
media_type = models.CharField(max_length=100)
|
|
active = models.BooleanField(default=True)
|
|
uid_key = models.BinaryField()
|
|
diversification_key = models.BinaryField()
|
|
|
|
objects = ScopedManager(organizer='organizer')
|
|
|
|
class Meta:
|
|
constraints = [
|
|
models.UniqueConstraint(
|
|
fields=["organizer", "media_type"],
|
|
condition=Q(active=True),
|
|
name="keyset_unique_active",
|
|
),
|
|
]
|