1 # This file is part of Wolne Lektury, licensed under GNU Affero GPLv3 or later.
2 # Copyright © Fundacja Wolne Lektury. See NOTICE for more information.
4 from django.conf import settings
5 from django.contrib.auth.decorators import login_required, permission_required
6 from django.db.models import Sum
7 from django.http import HttpResponse, HttpResponseRedirect, JsonResponse
8 from django.shortcuts import get_object_or_404, render
9 from django.urls import reverse
10 from django.utils.decorators import method_decorator
11 from django.views.decorators.cache import never_cache
12 from django.views.generic import FormView, CreateView, TemplateView, DetailView, UpdateView
13 from django.views import View
14 from .payu import POSS
15 from .payu import views as payu_views
16 from .forms import PayUCardTokenForm
19 from .helpers import get_active_schedule
20 from .payment_methods import recurring_payment_method
23 class ClubView(TemplateView):
24 template_name = 'club/index.html'
26 def get_context_data(self, *args, **kwargs):
27 ctx = super().get_context_data(*args, **kwargs)
28 ctx['active_menu_item'] = 'club'
33 @method_decorator(never_cache, name='dispatch')
34 class DonationStep1(UpdateView):
35 queryset = models.Schedule.objects.filter(payed_at=None)
36 form_class = forms.DonationStep1Form
37 slug_field = slug_url_kwarg = 'key'
38 template_name = 'club/donation_step1.html'
41 def get_context_data(self, **kwargs):
42 c = super().get_context_data(**kwargs)
43 c['club'] = models.Club.objects.first()
46 def get_success_url(self):
47 return reverse('donation_step2', args=[self.object.key])
50 @method_decorator(never_cache, name='dispatch')
51 class DonationStep2(UpdateView):
52 queryset = models.Schedule.objects.filter(payed_at=None)
53 form_class = forms.DonationStep2Form
54 slug_field = slug_url_kwarg = 'key'
55 template_name = 'club/donation_step2.html'
58 def get_context_data(self, **kwargs):
59 c = super().get_context_data(**kwargs)
60 c['club'] = models.Club.objects.first()
64 def set_monthly(request, key):
65 schedule = get_object_or_404(models.Schedule, payed_at=None, key=key)
67 schedule.monthly = True
68 schedule.save(update_fields=['monthly'])
70 "amount": schedule.amount,
71 "monthly": schedule.monthly,
75 class JoinView(CreateView):
76 form_class = forms.DonationStep1Form
77 template_name = 'club/donation_step1.html'
81 return models.Club.objects.first()
84 return self.request.GET.get('app')
86 def get(self, request):
87 if settings.CLUB_APP_HOST and self.is_app() and request.META['HTTP_HOST'] != settings.CLUB_APP_HOST:
88 return HttpResponseRedirect("https://" + settings.CLUB_APP_HOST + request.get_full_path())
91 request.session['from_app'] = True
92 elif request.session and 'from_app' in request.session:
93 del request.session['from_app']
95 return super().get(request)
97 def get_context_data(self, **kwargs):
98 c = super(JoinView, self).get_context_data(**kwargs)
99 c['membership'] = getattr(self.request.user, 'membership', None)
100 c['active_menu_item'] = 'club'
101 c['club'] = models.Club.objects.first()
103 c['ambassador'] = models.Ambassador.objects.all().order_by('?').first()
106 def get_initial(self):
108 if self.request.user.is_authenticated and self.request.user.email:
110 'email': self.request.user.email
113 def get_form_kwargs(self):
114 kwargs = super().get_form_kwargs()
115 kwargs['referer'] = self.request.META.get('HTTP_REFERER', '')
118 def form_valid(self, form):
119 retval = super(JoinView, self).form_valid(form)
120 if self.request.user.is_authenticated:
121 form.instance.membership, created = models.Membership.objects.get_or_create(user=self.request.user)
125 def get_success_url(self):
126 return reverse('donation_step2', args=[self.object.key])
129 @method_decorator(never_cache, name='dispatch')
130 class ScheduleView(DetailView):
131 queryset = models.Schedule.objects.exclude(email='')
132 slug_field = slug_url_kwarg = 'key'
133 template_name = 'club/schedule.html'
136 def get_template_names(self):
137 if not self.object.payed_at:
138 return 'club/donation_step3.html'
139 return 'club/schedule.html'
141 def get_context_data(self, *args, **kwargs):
142 ctx = super().get_context_data(*args, **kwargs)
143 ctx['active_menu_item'] = 'club'
146 def post(self, request, key):
147 schedule = self.get_object()
148 return HttpResponseRedirect(schedule.initiate_payment(request))
152 def claim(request, key):
153 schedule = models.Schedule.objects.get(key=key, membership=None)
154 schedule.membership, created = models.Membership.objects.get_or_create(user=request.user)
156 return HttpResponseRedirect(schedule.get_absolute_url())
159 def cancel(request, key):
160 schedule = models.Schedule.objects.get(key=key)
161 schedule.is_cancelled = True
163 return HttpResponseRedirect(schedule.get_absolute_url())
166 class PayUPayment(DetailView):
167 model = models.Schedule
168 slug_field = slug_url_kwarg = 'key'
170 def get(self, request, key):
171 schedule = self.get_object()
172 schedule.method = 'payu'
173 schedule.save(update_fields=['method'])
174 return HttpResponseRedirect(schedule.initiate_payment(request))
178 class PayURecPayment(payu_views.RecPayment):
179 form_class = PayUCardTokenForm
181 def get_schedule(self):
182 return get_object_or_404(models.Schedule, key=self.kwargs['key'])
185 pos_id = recurring_payment_method.pos_id
188 def get_success_url(self):
189 schedule = self.get_schedule()
190 schedule.method = 'payu-re'
191 schedule.save(update_fields=['method'])
192 return schedule.pay(self.request)
195 class PayUNotifyView(payu_views.NotifyView):
196 order_model = models.PayUOrder
199 class ScheduleThanksView(DetailView):
200 model = models.Schedule
201 template_name = 'club/donation_step4.html'
202 slug_field = slug_url_kwarg = 'key'
205 def get_context_data(self, *args, **kwargs):
206 ctx = super().get_context_data(*args, **kwargs)
207 ctx['active_menu_item'] = 'club'
211 class YearSummaryView(DetailView):
212 model = models.Schedule
213 slug_field = slug_url_kwarg = 'key'
214 template_name = 'club/year_summary.html'
216 def get_context_data(self, *args, **kwargs):
217 ctx = super().get_context_data(*args, **kwargs)
218 ctx['payments'] = models.PayUOrder.objects.filter(
220 completed_at__year=self.kwargs['year'],
221 schedule__email=self.object.email,
222 ).order_by('completed_at')
223 ctx['total_amount'] = ctx['payments'].aggregate(s=Sum('schedule__amount'))['s']
227 @permission_required('club.schedule_view')
228 def member_verify(request):
229 if request.method == 'POST':
230 emails = request.POST.get('emails').strip().split('\n')
231 rows = ['email;członek;nazwa użytkownika;aktywny;co najmniej do']
233 email = email.strip()
235 schedules = models.Schedule.objects.filter(email=email).exclude(payed_at=None)
236 if schedules.exists():
245 unames.add(s.membership.user.username)
246 if exp is None or s.expires_at > exp:
248 row.append(','.join(sorted(unames)))
249 row.append('tak' if akt else 'nie')
250 row.append(exp.date().isoformat())
253 rows.append(';'.join(row))
254 rows = '\n'.join(rows)
260 'club/member_verify.html',
267 @permission_required('club.schedule_view')
268 def receipt(request):
269 email = request.POST.get('email')
271 year = int(request.POST.get('year'))
273 return HttpResponse('no content')
275 receipt = models.PayUOrder.generate_receipt(email, year)
277 content, optout, payments = receipt
279 return HttpResponse('no content')
283 "Content-Type": "application/pdf",
284 "Content-Disposition": f'attachment; filename="wolnelektury-{year}-{email}.pdf"',