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 datetime import date, timedelta
5 from django.conf import settings
6 from django.contrib.auth.decorators import login_required, permission_required
7 from django.db.models import Sum
8 from django.http import HttpResponse, HttpResponseRedirect, JsonResponse
9 from django.shortcuts import get_object_or_404, render
10 from django.urls import reverse
11 from django.utils.decorators import method_decorator
12 from django.views.decorators.cache import never_cache
13 from django.views.generic import FormView, CreateView, TemplateView, DetailView, UpdateView
14 from django.views import View
15 from .payu import POSS
16 from .payu import views as payu_views
17 from .forms import PayUCardTokenForm
20 from .helpers import get_active_schedule
21 from .payment_methods import recurring_payment_method
24 class ClubView(TemplateView):
25 template_name = 'club/index.html'
27 def get_context_data(self, *args, **kwargs):
28 ctx = super().get_context_data(*args, **kwargs)
29 ctx['active_menu_item'] = 'club'
34 @method_decorator(never_cache, name='dispatch')
35 class DonationStep1(UpdateView):
36 queryset = models.Schedule.objects.filter(payed_at=None)
37 form_class = forms.DonationStep1Form
38 slug_field = slug_url_kwarg = 'key'
39 template_name = 'club/donation_step1.html'
42 def get_context_data(self, **kwargs):
43 c = super().get_context_data(**kwargs)
44 c['club'] = models.Club.objects.first()
47 def get_success_url(self):
48 return reverse('donation_step2', args=[self.object.key])
51 @method_decorator(never_cache, name='dispatch')
52 class DonationStep2(UpdateView):
53 queryset = models.Schedule.objects.filter(payed_at=None)
54 form_class = forms.DonationStep2Form
55 slug_field = slug_url_kwarg = 'key'
56 template_name = 'club/donation_step2.html'
59 def get_context_data(self, **kwargs):
60 c = super().get_context_data(**kwargs)
61 c['club'] = models.Club.objects.first()
65 def set_monthly(request, key):
66 schedule = get_object_or_404(models.Schedule, payed_at=None, key=key)
68 schedule.monthly = request.POST.get('monthly') == 'true'
69 schedule.save(update_fields=['monthly'])
71 "amount": schedule.amount,
72 "monthly": schedule.monthly,
76 class JoinView(CreateView):
77 form_class = forms.DonationStep1Form
78 template_name = 'club/donation_step1.html'
82 return models.Club.objects.first()
85 return self.request.GET.get('app')
87 def get(self, request):
88 if settings.CLUB_APP_HOST and self.is_app() and request.META['HTTP_HOST'] != settings.CLUB_APP_HOST:
89 return HttpResponseRedirect("https://" + settings.CLUB_APP_HOST + request.get_full_path())
92 request.session['from_app'] = True
93 elif request.session and 'from_app' in request.session:
94 del request.session['from_app']
96 return super().get(request)
98 def get_context_data(self, **kwargs):
99 c = super(JoinView, self).get_context_data(**kwargs)
100 c['membership'] = getattr(self.request.user, 'membership', None)
101 c['active_menu_item'] = 'club'
102 c['club'] = models.Club.objects.first()
104 c['ambassador'] = models.Ambassador.objects.all().order_by('?').first()
107 def get_initial(self):
109 if self.request.user.is_authenticated and self.request.user.email:
111 'email': self.request.user.email
114 def get_form_kwargs(self):
115 kwargs = super().get_form_kwargs()
116 kwargs['referer'] = self.request.META.get('HTTP_REFERER', '')
119 def form_valid(self, form):
120 retval = super(JoinView, self).form_valid(form)
121 if self.request.user.is_authenticated:
122 form.instance.membership, created = models.Membership.objects.get_or_create(user=self.request.user)
126 def get_success_url(self):
127 return reverse('donation_step2', args=[self.object.key])
130 @method_decorator(never_cache, name='dispatch')
131 class ScheduleView(DetailView):
132 queryset = models.Schedule.objects.exclude(email='')
133 slug_field = slug_url_kwarg = 'key'
134 template_name = 'club/schedule.html'
137 def get_template_names(self):
138 if not self.object.payed_at:
139 return 'club/donation_step3.html'
140 return 'club/schedule.html'
142 def get_context_data(self, *args, **kwargs):
143 ctx = super().get_context_data(*args, **kwargs)
144 ctx['active_menu_item'] = 'club'
147 def post(self, request, key):
148 schedule = self.get_object()
149 return HttpResponseRedirect(schedule.initiate_payment(request))
153 def claim(request, key):
154 schedule = models.Schedule.objects.get(key=key, membership=None)
155 schedule.membership, created = models.Membership.objects.get_or_create(user=request.user)
157 return HttpResponseRedirect(schedule.get_absolute_url())
160 def cancel(request, key):
161 schedule = models.Schedule.objects.get(key=key)
162 schedule.is_cancelled = True
164 return HttpResponseRedirect(schedule.get_absolute_url())
167 class PayUPayment(DetailView):
168 model = models.Schedule
169 slug_field = slug_url_kwarg = 'key'
171 def get(self, request, key):
172 schedule = self.get_object()
173 schedule.method = 'payu'
174 schedule.save(update_fields=['method'])
175 return HttpResponseRedirect(schedule.initiate_payment(request))
179 class PayURecPayment(payu_views.RecPayment):
180 form_class = PayUCardTokenForm
182 def get_schedule(self):
183 return get_object_or_404(models.Schedule, key=self.kwargs['key'])
186 pos_id = recurring_payment_method.pos_id
189 def get_success_url(self):
190 schedule = self.get_schedule()
191 schedule.method = 'payu-re'
192 schedule.save(update_fields=['method'])
193 return schedule.pay(self.request)
196 class PayUNotifyView(payu_views.NotifyView):
197 order_model = models.PayUOrder
200 class ScheduleThanksView(DetailView):
201 model = models.Schedule
202 template_name = 'club/donation_step4.html'
203 slug_field = slug_url_kwarg = 'key'
206 def get_context_data(self, *args, **kwargs):
207 ctx = super().get_context_data(*args, **kwargs)
208 ctx['active_menu_item'] = 'club'
212 class YearSummaryView(DetailView):
213 model = models.Schedule
214 slug_field = slug_url_kwarg = 'key'
215 template_name = 'club/year_summary.html'
217 def get_context_data(self, *args, **kwargs):
218 ctx = super().get_context_data(*args, **kwargs)
219 ctx['payments'] = models.PayUOrder.objects.filter(
221 completed_at__year=self.kwargs['year'],
222 schedule__email=self.object.email,
223 ).order_by('completed_at')
224 ctx['total_amount'] = ctx['payments'].aggregate(s=Sum('schedule__amount'))['s']
228 @permission_required('club.schedule_view')
229 def member_verify(request):
230 if request.method == 'POST':
231 emails = request.POST.get('emails').strip().split('\n')
232 rows = ['email;członek;nazwa użytkownika;aktywny;co najmniej do']
234 email = email.strip()
236 schedules = models.Schedule.objects.filter(email=email).exclude(payed_at=None)
237 if schedules.exists():
246 unames.add(s.membership.user.username)
247 if exp is None or s.expires_at > exp:
249 row.append(','.join(sorted(unames)))
250 row.append('tak' if akt else 'nie')
251 row.append(exp.date().isoformat())
254 rows.append(';'.join(row))
255 rows = '\n'.join(rows)
261 'club/member_verify.html',
268 @permission_required('club.schedule_view')
269 def receipt(request):
270 email = request.POST.get('email')
272 year = int(request.POST.get('year'))
274 return HttpResponse('no content')
276 receipt = models.PayUOrder.generate_receipt(email, year)
278 content, optout, payments = receipt
280 return HttpResponse('no content')
284 "Content-Type": "application/pdf",
285 "Content-Disposition": f'attachment; filename="wolnelektury-{year}-{email}.pdf"',
290 @permission_required('club.schedule_view')
295 start = today - timedelta(365)
296 for schedule in models.Schedule.objects.filter(
299 d = schedule.payed_at.date()
300 m = schedule.method.replace('-', '_')
301 acq.setdefault(d, {})
302 acq[d].setdefault(m, 0)
303 acq[d][m] += schedule.amount
305 for a in acq.values():
306 for m, v in a.items():
307 maxes.setdefault(m, 0)
315 for k, v in a.items():
316 a[k] = (v, 100 * v/(maxes[k] or 1))
317 days.append((d.isoformat(), a))
320 return render(request, 'club/stats.html',