548 lines
18 KiB
Python
548 lines
18 KiB
Python
import uuid
|
|
from datetime import datetime
|
|
from typing import Any
|
|
from urllib.parse import quote, urlencode
|
|
|
|
from django.conf import settings
|
|
from django.contrib import messages
|
|
from django.contrib.auth.decorators import permission_required
|
|
from django.contrib.auth.mixins import (
|
|
AccessMixin,
|
|
PermissionRequiredMixin,
|
|
)
|
|
from django.contrib.postgres.aggregates import StringAgg
|
|
from django.core import mail
|
|
from django.core.files.base import ContentFile
|
|
from django.db.models import OuterRef, Q, Subquery
|
|
from django.db.models.functions import TruncMonth, TruncYear
|
|
from django.http import HttpRequest, HttpResponse
|
|
from django.shortcuts import render
|
|
from django.template.loader import render_to_string
|
|
from django.utils import timezone
|
|
from django.utils.functional import cached_property
|
|
from django.views.generic import DetailView, ListView
|
|
from django.views.generic.dates import (
|
|
ArchiveIndexView,
|
|
MonthArchiveView,
|
|
YearArchiveView,
|
|
)
|
|
from django.views.generic.detail import BaseDetailView
|
|
from django.views.generic.edit import FormMixin, ProcessFormView
|
|
|
|
import django_filters
|
|
import django_tables2 as tables
|
|
import weasyprint
|
|
from dal import autocomplete
|
|
from django_filters.views import BaseFilterView
|
|
from django_sendfile import sendfile
|
|
from django_tables2 import A, SingleTableMixin
|
|
from django_tables2.export.views import ExportMixin
|
|
from django_weasyprint import WeasyTemplateResponseMixin
|
|
from django_weasyprint.utils import django_url_fetcher
|
|
|
|
from membershipworks.membershipworks_api import MembershipWorks
|
|
from membershipworks.tasks.scrape import scrape_event_details
|
|
|
|
from .forms import EventInvoiceForm
|
|
from .invoice_email import make_invoice_emails
|
|
from .models import EventAttendee, EventExt, EventInvoice, Member
|
|
from .tables import (
|
|
CurrentAndUpcomingEventTable,
|
|
EventAttendeeTable,
|
|
EventRegistrationsTable,
|
|
EventSummaryTable,
|
|
EventTable,
|
|
InvoiceTable,
|
|
MissingPaperworkTable,
|
|
UserEventTable,
|
|
)
|
|
|
|
|
|
class MemberAutocomplete(autocomplete.Select2QuerySetView):
|
|
model = Member
|
|
search_fields = ["account_name"]
|
|
|
|
def get_queryset(self):
|
|
if not self.request.user.has_perm("membershipworks.view_member"):
|
|
return Member.objects.none()
|
|
else:
|
|
return super().get_queryset()
|
|
|
|
|
|
@permission_required("membershipworks.view_eventext")
|
|
def upcoming_events_wordpress(request):
|
|
now = datetime.now()
|
|
|
|
membershipworks = MembershipWorks()
|
|
membershipworks.login(
|
|
settings.MEMBERSHIPWORKS_USERNAME, settings.MEMBERSHIPWORKS_PASSWORD
|
|
)
|
|
|
|
events = membershipworks.get_events_list(now)
|
|
if "error" in events:
|
|
messages.add_message(
|
|
request,
|
|
messages.ERROR,
|
|
f"MembershipWorks Error: {events['error']}",
|
|
)
|
|
# TODO: this should probably be an HTTP 500 response
|
|
return render(request, "base.dj.html")
|
|
|
|
ongoing_events = []
|
|
full_events = []
|
|
upcoming_events = []
|
|
for event in events["evt"]:
|
|
try:
|
|
# ignore hidden events
|
|
if event["cal"] == 0:
|
|
continue
|
|
event_details = membershipworks.get_event_by_eid(event["eid"])
|
|
|
|
# Convert timestamps to datetime objects
|
|
event_details["sdp_dt"] = datetime.fromtimestamp(event_details["sdp"])
|
|
event_details["edp_dt"] = datetime.fromtimestamp(event_details["edp"])
|
|
|
|
# registration has already ended
|
|
if (
|
|
"erd" in event_details
|
|
and datetime.fromtimestamp(event_details["erd"]) < now
|
|
):
|
|
ongoing_events.append(event_details)
|
|
# canceled events
|
|
elif event_details["cap"] == 0:
|
|
continue
|
|
# class is full
|
|
elif event_details["cnt"] >= event_details["cap"]:
|
|
full_events.append(event_details)
|
|
else:
|
|
upcoming_events.append(event_details)
|
|
|
|
except KeyError as e:
|
|
messages.add_message(
|
|
request,
|
|
messages.ERROR,
|
|
f"Event '{event.get('ttl')}' missing required property: '{e.args[0]}'",
|
|
)
|
|
# TODO: this should probably be an HTTP 500 response
|
|
return render(request, "base.dj.html")
|
|
|
|
context = {
|
|
"event_sections": [
|
|
{
|
|
"title": "Upcoming Events",
|
|
"blurb": "Events that are currently open for registration.",
|
|
"events": upcoming_events,
|
|
"truncate": False,
|
|
},
|
|
{
|
|
"title": "Just Missed",
|
|
"blurb": "These classes are currently full at time of writing. If you are interested, please check the event's page; spots occasionally open up. Keep an eye on this newsletter to see when these classes are offered again.",
|
|
"events": full_events,
|
|
"truncate": True,
|
|
},
|
|
{
|
|
"title": "Ongoing Events",
|
|
"blurb": "These events are ongoing. Registration is currently closed, but these events may be offered again in the future.",
|
|
"events": ongoing_events,
|
|
"truncate": True,
|
|
},
|
|
]
|
|
}
|
|
return render(request, "membershipworks/upcoming_events.dj.html", context)
|
|
|
|
|
|
class EventIndexReport(
|
|
ExportMixin, SingleTableMixin, PermissionRequiredMixin, ArchiveIndexView
|
|
):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventExt.objects.all()
|
|
date_field = "start"
|
|
template_name = "membershipworks/event_index_report.dj.html"
|
|
make_object_list = True
|
|
table_class = EventSummaryTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
export_name = "mw_events_index"
|
|
|
|
def get_table_data(self):
|
|
return (
|
|
super()
|
|
.get_table_data()
|
|
.with_financials()
|
|
.values(year=TruncYear("start"))
|
|
.summarize()
|
|
.order_by("year")
|
|
)
|
|
|
|
def get_table_kwargs(self):
|
|
year_column = tables.DateColumn(
|
|
"Y",
|
|
linkify=(
|
|
"membershipworks:event-year-report",
|
|
[A("year__year")],
|
|
),
|
|
)
|
|
return {
|
|
"sequence": ("year", "..."),
|
|
"extra_columns": (("year", year_column),),
|
|
}
|
|
|
|
|
|
class EventYearReport(
|
|
ExportMixin, SingleTableMixin, PermissionRequiredMixin, YearArchiveView
|
|
):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventExt.objects.all()
|
|
date_field = "start"
|
|
template_name = "membershipworks/event_year_report.dj.html"
|
|
make_object_list = True
|
|
table_class = EventSummaryTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
|
|
def get_table_data(self):
|
|
return (
|
|
super()
|
|
.get_table_data()
|
|
.with_financials()
|
|
.values(month=TruncMonth("start"))
|
|
.summarize()
|
|
.order_by("month")
|
|
)
|
|
|
|
def get_export_filename(self, export_format):
|
|
return f"mw_events_{self.get_year()}.{export_format}"
|
|
|
|
def get_table_kwargs(self):
|
|
month_column = tables.DateColumn(
|
|
"F Y",
|
|
linkify=(
|
|
"membershipworks:event-month-report",
|
|
[A("month__year"), A("month__month")],
|
|
),
|
|
)
|
|
return {
|
|
"sequence": ("month", "..."),
|
|
"extra_columns": (("month", month_column),),
|
|
}
|
|
|
|
|
|
class EventMonthReport(
|
|
ExportMixin, SingleTableMixin, PermissionRequiredMixin, MonthArchiveView
|
|
):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventExt.objects.all()
|
|
date_field = "start"
|
|
template_name = "membershipworks/event_month_report.dj.html"
|
|
table_class = EventTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
|
|
def get_table_data(self):
|
|
return (
|
|
super()
|
|
.get_table_data()
|
|
.select_related("category", "instructor", "instructor__member", "invoice")
|
|
.with_financials()
|
|
)
|
|
|
|
def get_export_filename(self, export_format):
|
|
return f"mw_events_{self.get_year()}-{self.get_month():02}.{export_format}"
|
|
|
|
|
|
class CurrentAndUpcomingEventView(SingleTableMixin, PermissionRequiredMixin, ListView):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventExt.objects.all()
|
|
date_field = "start"
|
|
template_name = "membershipworks/current_and_upcoming_events.dj.html"
|
|
table_class = CurrentAndUpcomingEventTable
|
|
|
|
def get_table_data(self):
|
|
return (
|
|
super()
|
|
.get_table_data()
|
|
.order_by("next_meeting_start")
|
|
.filter(end__gt=timezone.now())
|
|
.select_related("category", "instructor", "instructor__member")
|
|
.with_financials()
|
|
)
|
|
|
|
|
|
class UserEventView(SingleTableMixin, ListView):
|
|
model: type[EventExt] = EventExt
|
|
table_class = UserEventTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
template_name = "membershipworks/user_event_list.dj.html"
|
|
|
|
@cached_property
|
|
def member(self) -> Member | None:
|
|
return Member.from_user(self.request.user)
|
|
|
|
def get_queryset(self):
|
|
if self.member is None:
|
|
return self.model.objects.none()
|
|
else:
|
|
return super().get_queryset().filter(instructor__member=self.member)
|
|
|
|
def get_table_data(self):
|
|
return (
|
|
super()
|
|
.get_table_data()
|
|
.select_related("category", "invoice")
|
|
.with_financials()
|
|
)
|
|
|
|
def get_export_filename(self, export_format):
|
|
return f"my_events_{self.member.uid if self.member is not None else 'no-uid'}.{export_format}"
|
|
|
|
|
|
class EventDetailView(
|
|
SingleTableMixin, FormMixin, AccessMixin, DetailView, ProcessFormView
|
|
):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventExt.objects.with_financials().all()
|
|
pk_url_kwarg = "eid"
|
|
context_object_name = "event"
|
|
template_name = "membershipworks/event_detail.dj.html"
|
|
table_pagination = False
|
|
table_class = InvoiceTable
|
|
form_class = EventInvoiceForm
|
|
|
|
def render_to_response(
|
|
self, context: dict[str, Any], **response_kwargs: Any
|
|
) -> HttpResponse:
|
|
if self.request.user.has_perm(
|
|
self.permission_required
|
|
) or self.object.user_is_instructor(self.request.user):
|
|
return super().render_to_response(context, **response_kwargs)
|
|
else:
|
|
return self.handle_no_permission()
|
|
|
|
def display_instructor_version(self):
|
|
return (
|
|
self.request.method == "POST" # generating a PDF
|
|
or not self.request.user.has_perm(self.permission_required)
|
|
or self.request.GET.get("instructor_view")
|
|
)
|
|
|
|
def get_table_data(self):
|
|
if self.display_instructor_version():
|
|
return self.object.ticket_types.group_by_ticket_type()
|
|
else:
|
|
return self.object.ticket_types.all()
|
|
|
|
def get_table_kwargs(self):
|
|
kwargs = super().get_table_kwargs()
|
|
|
|
kwargs["event"] = self.object
|
|
if self.display_instructor_version():
|
|
kwargs["exclude"] = [
|
|
"list_price",
|
|
]
|
|
|
|
return kwargs
|
|
|
|
def get_success_url(self):
|
|
return self.request.build_absolute_uri()
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super().get_context_data(**kwargs)
|
|
context["user_is_instructor"] = self.object.user_is_instructor(
|
|
self.request.user
|
|
)
|
|
return context
|
|
|
|
def post(self, request, *args, **kwargs):
|
|
self.object = self.get_object()
|
|
return super().post(request, *args, **kwargs)
|
|
|
|
def get_form_kwargs(self):
|
|
kwargs = super().get_form_kwargs()
|
|
kwargs["event"] = self.object
|
|
kwargs["user"] = self.request.user
|
|
return kwargs
|
|
|
|
def form_valid(self, form: EventInvoiceForm):
|
|
self.object = self.get_object()
|
|
event = self.object
|
|
|
|
invoice_uuid = uuid.uuid4()
|
|
|
|
pdf_context = self.get_context_data(object=event)
|
|
pdf_context.update({"now": timezone.now(), "invoice_uuid": invoice_uuid})
|
|
weasy_html = weasyprint.HTML(
|
|
string=render_to_string(
|
|
"membershipworks/event_invoice_pdf.dj.html",
|
|
context=pdf_context,
|
|
request=self.request,
|
|
),
|
|
url_fetcher=django_url_fetcher,
|
|
base_url="file://",
|
|
)
|
|
pdf = weasy_html.write_pdf()
|
|
# the result will be None only if a target was provided
|
|
assert pdf is not None
|
|
|
|
# NOTE: this is only saved AFTER the emails are successfully sent
|
|
invoice = EventInvoice(
|
|
uuid=invoice_uuid,
|
|
event=event,
|
|
date_submitted=pdf_context["now"],
|
|
amount=event.total_due_to_instructor,
|
|
)
|
|
|
|
emails = make_invoice_emails(
|
|
invoice, pdf, self.request.build_absolute_uri(event.get_absolute_url())
|
|
)
|
|
with mail.get_connection() as conn:
|
|
conn.send_messages(emails)
|
|
|
|
# this also saves the invoice object
|
|
invoice.pdf.save(f"{event.eid}.pdf", ContentFile(pdf))
|
|
|
|
messages.success(
|
|
self.request,
|
|
"Created Invoice! You should receive a confirmation email shortly.",
|
|
)
|
|
|
|
return super().form_valid(form)
|
|
|
|
|
|
class EventInvoicePDFPreviewView(WeasyTemplateResponseMixin, EventDetailView):
|
|
template_name = "membershipworks/event_invoice_pdf.dj.html"
|
|
pdf_attachment = False
|
|
|
|
def display_instructor_version(self):
|
|
return True
|
|
|
|
def get_pdf_filename(self):
|
|
return f"event-invoice_{self.object.pk}.pdf"
|
|
|
|
def get_context_data(self, **kwargs):
|
|
context = super().get_context_data(**kwargs)
|
|
context.update(
|
|
{
|
|
"now": timezone.now(),
|
|
"invoice_uuid": "00000000-0000-0000-0000-000000000000",
|
|
"preview": True,
|
|
}
|
|
)
|
|
return context
|
|
|
|
|
|
class EventInvoicePDFView(AccessMixin, BaseDetailView):
|
|
model = EventInvoice
|
|
pk_url_kwarg = "uuid"
|
|
|
|
def get(self, request: HttpRequest, *args: Any, **kwargs: Any) -> HttpResponse:
|
|
invoice = self.get_object()
|
|
if request.user.has_perm(
|
|
"membershipworks.view_eventinvoice"
|
|
) or invoice.event.user_is_instructor(request.user):
|
|
return sendfile(request, invoice.pdf.path, mimetype="application/pdf")
|
|
|
|
else:
|
|
return self.handle_no_permission()
|
|
|
|
|
|
class EventRegistrationsView(ExportMixin, SingleTableMixin, AccessMixin, DetailView):
|
|
permission_required = "membershipworks.view_eventext"
|
|
model = EventExt
|
|
pk_url_kwarg = "eid"
|
|
context_object_name = "event"
|
|
template_name = "membershipworks/event_registrations.dj.html"
|
|
table_class = EventRegistrationsTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
|
|
def render_to_response(
|
|
self, context: dict[str, Any], **response_kwargs: Any
|
|
) -> HttpResponse:
|
|
if "refresh" in self.request.GET:
|
|
scrape_event_details([self.object])
|
|
|
|
if self.request.user.has_perm(
|
|
self.permission_required
|
|
) or self.object.user_is_instructor(self.request.user):
|
|
return super().render_to_response(context, **response_kwargs)
|
|
else:
|
|
return self.handle_no_permission()
|
|
|
|
def get_context_data(self, **kwargs: Any) -> dict[str, Any]:
|
|
context_data = super().get_context_data(**kwargs)
|
|
|
|
context_data["email_link"] = "mailto:?" + urlencode(
|
|
{
|
|
"subject": f"[CMS Event] {self.object.title}",
|
|
"bcc": ",".join(
|
|
mail.message.sanitize_address(
|
|
(reg["Full name"], reg["Email"]), settings.DEFAULT_CHARSET
|
|
)
|
|
for reg in self.object.registrations
|
|
if any(
|
|
int(v) > 0 for k, v in reg.items() if k.startswith("Ticket: ")
|
|
)
|
|
),
|
|
},
|
|
quote_via=quote,
|
|
)
|
|
|
|
return context_data
|
|
|
|
def get_table_data(self):
|
|
return self.object.registrations
|
|
|
|
|
|
class EventAttendeeFilters(django_filters.FilterSet):
|
|
new_since = django_filters.DateFilter(
|
|
field_name="event__start", method="filter_new_since"
|
|
)
|
|
|
|
def filter_new_since(self, queryset, name, value):
|
|
return queryset.filter(**{f"{name}__gte": value}).exclude(
|
|
email__in=Subquery(
|
|
queryset.filter(**{f"{name}__lt": value}).values("email")
|
|
)
|
|
)
|
|
|
|
|
|
class EventAttendeeListView(
|
|
BaseFilterView, ExportMixin, SingleTableMixin, PermissionRequiredMixin, ListView
|
|
):
|
|
permission_required = "membershipworks.view_eventext"
|
|
queryset = EventAttendee.objects.all()
|
|
table_class = EventAttendeeTable
|
|
template_name = "membershipworks/eventattendee_list.dj.html"
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
filterset_class = EventAttendeeFilters
|
|
|
|
def get_table_data(self):
|
|
return super().get_table_data().values("name", "email").distinct()
|
|
|
|
|
|
class MissingPaperworkReport(
|
|
ExportMixin,
|
|
SingleTableMixin,
|
|
PermissionRequiredMixin,
|
|
ListView,
|
|
):
|
|
model = Member
|
|
permission_required = "membershipworks.view_member"
|
|
template_name = "membershipworks/missing_paperwork_report.dj.html"
|
|
table_class = MissingPaperworkTable
|
|
export_formats = ("csv", "xlsx", "ods")
|
|
|
|
def get_queryset(self):
|
|
qs = super().get_queryset()
|
|
return (
|
|
qs.with_is_active()
|
|
.filter(
|
|
Q(membership_agreement_signed_and_on_file_date__isnull=True)
|
|
| Q(waiver_form_signed_and_on_file_date__isnull=True),
|
|
is_active=True,
|
|
)
|
|
.annotate(
|
|
membership=Subquery(
|
|
qs.filter(
|
|
pk=OuterRef("pk"), flags__type__in=("level", "addon")
|
|
).values(m=StringAgg("flags__name", ", "))
|
|
),
|
|
)
|
|
)
|