from __future__ import annotations from django.core import signing from django.http import Http404, JsonResponse from django.shortcuts import get_object_or_404, redirect from django.views import View from apps.newsletter.forms import SubscriptionForm from apps.newsletter.models import NewsletterSubscription from apps.newsletter.services import ProviderSyncError, get_provider_service CONFIRMATION_TOKEN_MAX_AGE_SECONDS = 60 * 60 * 24 * 2 class SubscribeView(View): def post(self, request): form = SubscriptionForm(request.POST) if not form.is_valid(): return JsonResponse({"status": "error", "field": "email"}, status=400) if form.cleaned_data.get("honeypot"): return JsonResponse({"status": "ok"}) email = form.cleaned_data["email"] source = form.cleaned_data.get("source") or "unknown" NewsletterSubscription.objects.get_or_create(email=email, defaults={"source": source}) return JsonResponse({"status": "ok"}) class ConfirmView(View): def get(self, request, token: str): try: email = signing.loads( token, salt="newsletter-confirm", max_age=CONFIRMATION_TOKEN_MAX_AGE_SECONDS, ) except signing.BadSignature as exc: raise Http404 from exc subscription = get_object_or_404(NewsletterSubscription, email=email) subscription.confirmed = True subscription.save(update_fields=["confirmed"]) service = get_provider_service() try: service.sync(subscription) except ProviderSyncError: pass return redirect("/") def confirmation_token(email: str) -> str: return signing.dumps(email, salt="newsletter-confirm")