From d7a58ed907418d0244415b1c0986f22b20f76dff Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sat, 5 Oct 2024 17:35:02 +0200 Subject: [PATCH 01/10] Extend tickets with tags and better admin --- wafer/tickets/admin.py | 42 ++++++++++++++++++++++++++++++++++++++--- wafer/tickets/models.py | 20 ++++++++++++++++++++ 2 files changed, 59 insertions(+), 3 deletions(-) diff --git a/wafer/tickets/admin.py b/wafer/tickets/admin.py index 4205224c..eaaaf60c 100644 --- a/wafer/tickets/admin.py +++ b/wafer/tickets/admin.py @@ -1,6 +1,42 @@ from django.contrib import admin +from django.utils.translation import gettext_lazy as _ +from reversion.admin import VersionAdmin -from wafer.tickets.models import Ticket, TicketType +from wafer.tickets.models import Ticket, TicketType, TicketTypeTag -admin.site.register(Ticket) -admin.site.register(TicketType) + +class ClaimedFilter(admin.SimpleListFilter): + title = _('ticket claimed') + parameter_name = 'claimed' + + def lookups(self, request, model_admins): + return ( + ('yes', _('Ticket has been claimed')), + ('no', _('Ticket is unclaimed')) + ) + + def queryset(self, request, queryset): + if self.value() == 'yes': + return queryset.filter(user__isnull=False) + elif self.value() == 'no': + return queryset.filter(user__isnull=True) + return queryset + + +class TicketTypeTagAdmin(VersionAdmin): + pass + + +# We don't use the versioned admin here, as these are usually created and +# updated by external triggers and we don't currently version that +class TicketTypeAdmin(admin.ModelAdmin): + list_display = ('name', 'get_tags', 'get_count') + + +class TicketAdmin(admin.ModelAdmin): + list_filter = (ClaimedFilter, 'type') + + +admin.site.register(Ticket, TicketAdmin) +admin.site.register(TicketType, TicketTypeAdmin) +admin.site.register(TicketTypeTag, TicketTypeTagAdmin) diff --git a/wafer/tickets/models.py b/wafer/tickets/models.py index 1686898c..7f9ae69b 100644 --- a/wafer/tickets/models.py +++ b/wafer/tickets/models.py @@ -2,15 +2,35 @@ from django.conf import settings +class TicketTypeTag(models.Model): + """Tags that can be added to a TicketType. + + For example 'online, 'sponsor'""" + + MAX_NAME_LENGTH = 75 + name = models.CharField(max_length=MAX_NAME_LENGTH) + + def __str__(self): + return self.name + + class TicketType(models.Model): MAX_NAME_LENGTH = 255 name = models.CharField(max_length=MAX_NAME_LENGTH) + tags = models.ManyToManyField(TicketTypeTag) + def __str__(self): return self.name + def get_count(self): + return 0 + + def get_tags(self): + return '' + class Ticket(models.Model): barcode = models.IntegerField(primary_key=True) From 518f2ab85b248c27cdfb72dbc5f7b1972eec553c Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sat, 5 Oct 2024 22:22:50 +0200 Subject: [PATCH 02/10] Add migration --- .../0004_tickettypetag_tickettype_tags.py | 32 +++++++++++++++++++ 1 file changed, 32 insertions(+) create mode 100644 wafer/tickets/migrations/0004_tickettypetag_tickettype_tags.py diff --git a/wafer/tickets/migrations/0004_tickettypetag_tickettype_tags.py b/wafer/tickets/migrations/0004_tickettypetag_tickettype_tags.py new file mode 100644 index 00000000..a78b574a --- /dev/null +++ b/wafer/tickets/migrations/0004_tickettypetag_tickettype_tags.py @@ -0,0 +1,32 @@ +# Generated by Django 5.0.9 on 2024-10-05 15:38 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + dependencies = [ + ("tickets", "0003_longer_email_field"), + ] + + operations = [ + migrations.CreateModel( + name="TicketTypeTag", + fields=[ + ( + "id", + models.AutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("name", models.CharField(max_length=75)), + ], + ), + migrations.AddField( + model_name="tickettype", + name="tags", + field=models.ManyToManyField(to="tickets.tickettypetag"), + ), + ] From b5133175cde85224cc6b62f86f7ffc1447f5d2bc Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 10:40:02 +0200 Subject: [PATCH 03/10] Add tag output --- wafer/tickets/models.py | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/wafer/tickets/models.py b/wafer/tickets/models.py index 7f9ae69b..84b59309 100644 --- a/wafer/tickets/models.py +++ b/wafer/tickets/models.py @@ -25,11 +25,10 @@ class TicketType(models.Model): def __str__(self): return self.name - def get_count(self): - return 0 - def get_tags(self): - return '' + return ', '.join([x.name for x in self.tags.all()]) + + get_tags.short_description = 'tags' class Ticket(models.Model): From 4900c62e5f6a5581e00ad414fad81fb74d285371 Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 10:40:26 +0200 Subject: [PATCH 04/10] Fix up ticket type admin --- wafer/tickets/admin.py | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/wafer/tickets/admin.py b/wafer/tickets/admin.py index eaaaf60c..02cbf349 100644 --- a/wafer/tickets/admin.py +++ b/wafer/tickets/admin.py @@ -1,4 +1,5 @@ from django.contrib import admin +from django.db import models from django.utils.translation import gettext_lazy as _ from reversion.admin import VersionAdmin @@ -22,7 +23,6 @@ def queryset(self, request, queryset): return queryset.filter(user__isnull=True) return queryset - class TicketTypeTagAdmin(VersionAdmin): pass @@ -30,7 +30,21 @@ class TicketTypeTagAdmin(VersionAdmin): # We don't use the versioned admin here, as these are usually created and # updated by external triggers and we don't currently version that class TicketTypeAdmin(admin.ModelAdmin): - list_display = ('name', 'get_tags', 'get_count') + list_display = ('name', 'get_tags', 'get_ticket_count') + list_filter = ('tags',) + + def get_queryset(self, request): + qs = super().get_queryset(request) + qs = qs.annotate( + ticket_count_annotation=models.Count('ticket', distinct=True), + ) + return qs + + def get_ticket_count(self, obj): + return obj.ticket_count_annotation + + get_ticket_count.short_description = 'total purchased' + get_ticket_count.admin_order_field = 'ticket_count_annotation' class TicketAdmin(admin.ModelAdmin): From f75d44d5a73afa0df140ebda80716ca9fd4f8677 Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 12:20:19 +0200 Subject: [PATCH 05/10] Add some minimal tests on the new stuff --- wafer/tickets/tests/test_admin_helpers.py | 66 +++++++++++++++++++++++ 1 file changed, 66 insertions(+) create mode 100644 wafer/tickets/tests/test_admin_helpers.py diff --git a/wafer/tickets/tests/test_admin_helpers.py b/wafer/tickets/tests/test_admin_helpers.py new file mode 100644 index 00000000..f49d1b06 --- /dev/null +++ b/wafer/tickets/tests/test_admin_helpers.py @@ -0,0 +1,66 @@ +from django.contrib.admin.sites import AdminSite + +from django.test import TestCase, RequestFactory + +from wafer.tickets.models import TicketType, TicketTypeTag, Ticket +from wafer.tickets.admin import TicketTypeAdmin + +from wafer.tests.utils import create_user + +class TicketTypeAdminTests(TestCase): + + def setUp(self): + self.user_emails = ["user%d@example.com" % num for num in range(5)] + # create some tags + self.tag_z = TicketTypeTag.objects.create(name='tag z') + self.tag_a = TicketTypeTag.objects.create(name='tag a') + self.tag_d = TicketTypeTag.objects.create(name='tag d') + # Create some type + self.type_sponsor = TicketType.objects.create(name='Sponsor') + self.type_student = TicketType.objects.create(name='Student') + self.admin_model = TicketTypeAdmin(model=TicketType, admin_site=AdminSite()) + + self.admin_user = create_user('ticket_admin', superuser=True) + self.request_factory = RequestFactory() + + + def test_tags(self): + """Test that get_tags works as expected""" + # Add tags + self.type_sponsor.tags.add(self.tag_z) + self.type_sponsor.tags.add(self.tag_a) + + self.type_student.tags.add(self.tag_d) + self.type_student.tags.add(self.tag_z) + self.type_student.tags.add(self.tag_a) + + self.assertEqual(self.type_sponsor.get_tags(), 'tag a, tag z') + self.assertEqual(self.type_student.get_tags(), 'tag a, tag d, tag z') + + def test_ticket_counts(self): + """Test the get_count query on the admin model""" + + # Buy some tickets + Ticket.objects.create(email=self.user_emails[0], type=self.type_sponsor) + Ticket.objects.create(email=self.user_emails[1], type=self.type_sponsor) + Ticket.objects.create(email=self.user_emails[2], type=self.type_sponsor) + + request = self.request_factory.get("/") + request.user = self.admin_user + qs = self.admin_model.get_queryset(request) + self.assertEqual(self.admin_model.get_ticket_count(qs[0]), 3) + self.assertEqual(self.admin_model.get_ticket_count(qs[1]), 0) + + Ticket.objects.create(email=self.user_emails[3], type=self.type_student) + ticket5 = Ticket.objects.create(email=self.user_emails[4], type=self.type_student, + barcode='1111') + + qs = self.admin_model.get_queryset(request) + self.assertEqual(self.admin_model.get_ticket_count(qs[0]), 3) + self.assertEqual(self.admin_model.get_ticket_count(qs[1]), 2) + + ticket5.delete() + + qs = self.admin_model.get_queryset(request) + self.assertEqual(self.admin_model.get_ticket_count(qs[0]), 3) + self.assertEqual(self.admin_model.get_ticket_count(qs[1]), 1) From 61fe01aedcc1f44da74994bfc5e9fb6afd9b41a8 Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 12:25:43 +0200 Subject: [PATCH 06/10] Update pot file --- wafer/locale/django.pot | 21 ++++++++++++++------- 1 file changed, 14 insertions(+), 7 deletions(-) diff --git a/wafer/locale/django.pot b/wafer/locale/django.pot index eb07516c..e906b242 100644 --- a/wafer/locale/django.pot +++ b/wafer/locale/django.pot @@ -8,7 +8,7 @@ msgid "" msgstr "" "Project-Id-Version: PACKAGE VERSION\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2024-10-05 20:23+0000\n" +"POT-Creation-Date: 2024-10-06 10:25+0000\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "Last-Translator: FULL NAME \n" "Language-Team: LANGUAGE \n" @@ -190,12 +190,7 @@ msgstr "" msgid "No previous versions to compare to" msgstr "" -#: wafer/pages/templates/wafer.pages/page_form.html:5 -#, python-format -msgid "Editing Page: %(page)s - %(WAFER_CONFERENCE_NAME)s" -msgstr "" - -#: wafer/pages/templates/wafer.pages/page_form.html:12 +#: wafer/pages/templates/wafer.pages/page_form.html:11 msgid "Edit Page" msgstr "" @@ -1237,6 +1232,18 @@ msgstr "" msgid "Sign up / Log In" msgstr "" +#: wafer/tickets/admin.py:10 +msgid "ticket claimed" +msgstr "" + +#: wafer/tickets/admin.py:15 +msgid "Ticket has been claimed" +msgstr "" + +#: wafer/tickets/admin.py:16 +msgid "Ticket is unclaimed" +msgstr "" + #: wafer/tickets/forms.py:18 msgid "Claim" msgstr "" From 4a32d9d76b714b7841ff66dddbb9354376e5ea5c Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 12:27:27 +0200 Subject: [PATCH 07/10] add missing barcodes --- wafer/tickets/tests/test_admin_helpers.py | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/wafer/tickets/tests/test_admin_helpers.py b/wafer/tickets/tests/test_admin_helpers.py index f49d1b06..1a77b881 100644 --- a/wafer/tickets/tests/test_admin_helpers.py +++ b/wafer/tickets/tests/test_admin_helpers.py @@ -41,9 +41,12 @@ def test_ticket_counts(self): """Test the get_count query on the admin model""" # Buy some tickets - Ticket.objects.create(email=self.user_emails[0], type=self.type_sponsor) - Ticket.objects.create(email=self.user_emails[1], type=self.type_sponsor) - Ticket.objects.create(email=self.user_emails[2], type=self.type_sponsor) + Ticket.objects.create(email=self.user_emails[0], type=self.type_sponsor, + barcode='1234') + Ticket.objects.create(email=self.user_emails[1], type=self.type_sponsor, + barcode='2345') + Ticket.objects.create(email=self.user_emails[2], type=self.type_sponsor, + barcode='3456') request = self.request_factory.get("/") request.user = self.admin_user @@ -51,7 +54,8 @@ def test_ticket_counts(self): self.assertEqual(self.admin_model.get_ticket_count(qs[0]), 3) self.assertEqual(self.admin_model.get_ticket_count(qs[1]), 0) - Ticket.objects.create(email=self.user_emails[3], type=self.type_student) + Ticket.objects.create(email=self.user_emails[3], type=self.type_student, + barcode='4321') ticket5 = Ticket.objects.create(email=self.user_emails[4], type=self.type_student, barcode='1111') From 7b89c946007c93be9bb6b007a636239c2020cddf Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 12:29:07 +0200 Subject: [PATCH 08/10] Sort tag names in admin interface for predictable results --- wafer/tickets/models.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/wafer/tickets/models.py b/wafer/tickets/models.py index 84b59309..5aefd9d5 100644 --- a/wafer/tickets/models.py +++ b/wafer/tickets/models.py @@ -26,7 +26,7 @@ def __str__(self): return self.name def get_tags(self): - return ', '.join([x.name for x in self.tags.all()]) + return ', '.join([x.name for x in self.tags.all().order_by('name')]) get_tags.short_description = 'tags' From f20fbd4087bcc09aec199d61e6e2bbfaa00ab1c0 Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Sun, 6 Oct 2024 12:31:34 +0200 Subject: [PATCH 09/10] Fix pot file merge error --- wafer/locale/django.pot | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/wafer/locale/django.pot b/wafer/locale/django.pot index e906b242..136a7923 100644 --- a/wafer/locale/django.pot +++ b/wafer/locale/django.pot @@ -190,7 +190,12 @@ msgstr "" msgid "No previous versions to compare to" msgstr "" -#: wafer/pages/templates/wafer.pages/page_form.html:11 +#: wafer/pages/templates/wafer.pages/page_form.html:5 +#, python-format +msgid "Editing Page: %(page)s - %(WAFER_CONFERENCE_NAME)s" +msgstr "" + +#: wafer/pages/templates/wafer.pages/page_form.html:12 msgid "Edit Page" msgstr "" From f507636a56476584de33bc8fa92cf8e6d546ffe0 Mon Sep 17 00:00:00 2001 From: Neil Muller Date: Mon, 7 Oct 2024 15:47:10 +0200 Subject: [PATCH 10/10] Update wafer/tickets/models.py Co-authored-by: Simon Cross --- wafer/tickets/models.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/wafer/tickets/models.py b/wafer/tickets/models.py index 5aefd9d5..1eef945e 100644 --- a/wafer/tickets/models.py +++ b/wafer/tickets/models.py @@ -5,7 +5,8 @@ class TicketTypeTag(models.Model): """Tags that can be added to a TicketType. - For example 'online, 'sponsor'""" + For example 'online', 'sponsor'. + """ MAX_NAME_LENGTH = 75 name = models.CharField(max_length=MAX_NAME_LENGTH)