mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-31 07:49:57 +01:00 
			
		
		
		
	Compare commits
	
		
			14 Commits
		
	
	
		
			d3e832d23a
			...
			summary_no
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 5a0fe7a6f0 | ||
|  | eda8460014 | ||
|  | 15c71ad31a | ||
|  | b8f81048a5 | ||
|  | af819f45a1 | ||
|  | 076d065ffa | ||
|  | 2da77d9c17 | ||
|  | 01584d6330 | ||
|  | 4c0a5922c4 | ||
|  | f90b28fc7c | ||
|  | 925e0f26f5 | ||
|  | c912383f86 | ||
|  | 32830e43fd | ||
|  | 11c6a6fa7a | 
| @@ -17,7 +17,8 @@ from django.utils.translation import gettext_lazy as _ | ||||
| from django.views import View | ||||
| from django.views.decorators.cache import cache_page | ||||
| from django.views.generic import DetailView, TemplateView, UpdateView | ||||
| from django_tables2.views import SingleTableView | ||||
| from django.views.generic.list import ListView | ||||
| from django_tables2.views import MultiTableMixin | ||||
| from note.models import Alias, NoteSpecial, NoteUser | ||||
| from permission.backends import PermissionBackend | ||||
| from permission.views import ProtectQuerysetMixin, ProtectedCreateView | ||||
| @@ -57,27 +58,40 @@ class ActivityCreateView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|         return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk}) | ||||
|  | ||||
|  | ||||
| class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView): | ||||
| class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView): | ||||
|     """ | ||||
|     Displays all Activities, and classify if they are on-going or upcoming ones. | ||||
|     """ | ||||
|     model = Activity | ||||
|     table_class = ActivityTable | ||||
|     ordering = ('-date_start',) | ||||
|     tables = [ActivityTable, ActivityTable] | ||||
|     extra_context = {"title": _("Activities")} | ||||
|  | ||||
|     def get_queryset(self, **kwargs): | ||||
|         return super().get_queryset(**kwargs).distinct() | ||||
|  | ||||
|     def get_tables(self): | ||||
|         tables = super().get_tables() | ||||
|  | ||||
|         tables[0].prefix = "all-" | ||||
|         tables[1].prefix = "upcoming-" | ||||
|         return tables | ||||
|  | ||||
|     def get_tables_data(self): | ||||
|         # first table = all activities, second table = upcoming | ||||
|         return [ | ||||
|             self.get_queryset().order_by("-date_start"), | ||||
|             Activity.objects.filter(date_end__gt=timezone.now()) | ||||
|                             .filter(PermissionBackend.filter_queryset(self.request, Activity, "view")) | ||||
|                             .distinct() | ||||
|                             .order_by("date_start") | ||||
|         ] | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = super().get_context_data(**kwargs) | ||||
|  | ||||
|         upcoming_activities = Activity.objects.filter(date_end__gt=timezone.now()) | ||||
|         context['upcoming'] = ActivityTable( | ||||
|             data=upcoming_activities.filter(PermissionBackend.filter_queryset(self.request, Activity, "view")), | ||||
|             prefix='upcoming-', | ||||
|             order_by='date_start', | ||||
|         ) | ||||
|         tables = context["tables"] | ||||
|         for name, table in zip(["table", "upcoming"], tables): | ||||
|             context[name] = table | ||||
|  | ||||
|         started_activities = self.get_queryset().filter(open=True, valid=True).distinct().all() | ||||
|         context["started_activities"] = started_activities | ||||
|   | ||||
| @@ -138,6 +138,9 @@ class ImageForm(forms.Form): | ||||
|  | ||||
|         return cleaned_data | ||||
|  | ||||
|     def is_valid(self): | ||||
|         return super().is_valid() or super().clean().get('image') is None | ||||
|  | ||||
|  | ||||
| class ClubForm(forms.ModelForm): | ||||
|     def clean(self): | ||||
| @@ -151,7 +154,7 @@ class ClubForm(forms.ModelForm): | ||||
|  | ||||
|     class Meta: | ||||
|         model = Club | ||||
|         fields = '__all__' | ||||
|         exclude = ("add_registration_form",) | ||||
|         widgets = { | ||||
|             "membership_fee_paid": AmountInput(), | ||||
|             "membership_fee_unpaid": AmountInput(), | ||||
|   | ||||
							
								
								
									
										18
									
								
								apps/member/migrations/0012_club_add_registration_form.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/member/migrations/0012_club_add_registration_form.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Generated by Django 2.2.28 on 2024-07-15 09:24 | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ('member', '0011_profile_vss_charter_read'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AddField( | ||||
|             model_name='club', | ||||
|             name='add_registration_form', | ||||
|             field=models.BooleanField(default=False, verbose_name='add to registration form'), | ||||
|         ), | ||||
|     ] | ||||
							
								
								
									
										18
									
								
								apps/member/migrations/0013_auto_20240807_1409.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/member/migrations/0013_auto_20240807_1409.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Generated by Django 2.2.28 on 2024-08-07 12:09 | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ('member', '0012_club_add_registration_form'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AlterField( | ||||
|             model_name='profile', | ||||
|             name='promotion', | ||||
|             field=models.PositiveSmallIntegerField(default=2024, help_text='Year of entry to the school (None if not ENS student)', null=True, verbose_name='promotion'), | ||||
|         ), | ||||
|     ] | ||||
| @@ -259,6 +259,11 @@ class Club(models.Model): | ||||
|         help_text=_('Maximal date of a membership, after which members must renew it.'), | ||||
|     ) | ||||
|  | ||||
|     add_registration_form = models.BooleanField( | ||||
|         verbose_name=_("add to registration form"), | ||||
|         default=False, | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("club") | ||||
|         verbose_name_plural = _("clubs") | ||||
|   | ||||
| @@ -14,6 +14,9 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
|       <form method="post" enctype="multipart/form-data" id="formUpload"> | ||||
|         {% csrf_token %} | ||||
|         {{ form |crispy }} | ||||
|         {% if user.note.display_image != "pic/default.png" %} | ||||
|           <input type="submit" class="btn btn-primary" value="{% trans "Remove" %}"> | ||||
|         {% endif %} | ||||
|       </form> | ||||
|     </div> | ||||
|     <!-- MODAL TO CROP THE IMAGE --> | ||||
|   | ||||
| @@ -326,12 +326,15 @@ class PictureUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, Det | ||||
|         """Save image to note""" | ||||
|         image = form.cleaned_data['image'] | ||||
|  | ||||
|         # Rename as a PNG or GIF | ||||
|         extension = image.name.split(".")[-1] | ||||
|         if extension == "gif": | ||||
|             image.name = "{}_pic.gif".format(self.object.note.pk) | ||||
|         if image is None: | ||||
|             image = "pic/default.png" | ||||
|         else: | ||||
|             image.name = "{}_pic.png".format(self.object.note.pk) | ||||
|             # Rename as a PNG or GIF | ||||
|             extension = image.name.split(".")[-1] | ||||
|             if extension == "gif": | ||||
|                 image.name = "{}_pic.gif".format(self.object.note.pk) | ||||
|             else: | ||||
|                 image.name = "{}_pic.png".format(self.object.note.pk) | ||||
|  | ||||
|         # Save | ||||
|         self.object.note.display_image = image | ||||
|   | ||||
| @@ -23,7 +23,7 @@ | ||||
|  | ||||
| <p> | ||||
|     Par ailleurs, le BDE ne sert pas d'alcool aux adhérents dont le solde | ||||
|     est inférieur à 0 € depuis plus de 24h. | ||||
|     est inférieur à 0 €. | ||||
| </p> | ||||
|  | ||||
| <p> | ||||
|   | ||||
							
								
								
									
										65
									
								
								apps/note/templates/note/mails/summary_notes_report.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								apps/note/templates/note/mails/summary_notes_report.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| {% load pretty_money %} | ||||
| {% load i18n %} | ||||
|  | ||||
| <!DOCTYPE html> | ||||
| <html lang="fr"> | ||||
| <head> | ||||
|     <meta charset="UTF-8"> | ||||
|     <title>[Note Kfet] Récapitulatif de trésorerie</title> | ||||
| </head> | ||||
| <body> | ||||
| <h1> | ||||
|   Récapitulatif de trésorerie au {{ summary.date|date:"d/m/Y" }} à {{ summary.date|date:"H:i:s" }} : | ||||
| </h1> | ||||
|  | ||||
| <h2> | ||||
|   Tous les utilisateur⋅rices : | ||||
| </h2> | ||||
| <ul> | ||||
|   <li>Positifs : {{ summary.total_positive_user }} soit {{ summary.balance_positive_user / 100 }} €</li> | ||||
|   <li>Neutres : {{ summary.total_zero_user }}</li> | ||||
|   <li>Négatifs : {{ summary.total_negative_user }} soit {{ summary.balance_negative_user / 100 }} €</li> | ||||
| </ul> | ||||
|  | ||||
| <h2> | ||||
|   Les {{ summary.total_positive_user_bde + summary.total_zero_user_bde + summary.total_negative_user_bde }} adhérent⋅es BDE : | ||||
| </h2> | ||||
| <ul> | ||||
|   <li>Positifs : {{ summary.total_positive_user_bde }} soit {{ summary.balance_positive_user_bde / 100 }} €</li> | ||||
|   <li>Neutres : {{ summary.total_zero_user_bde }}</li> | ||||
|   <li>Négatifs : {{ summary.total_negative_user_bde }} soit {{ summary.balance_negative_user_bde / 100 }} €</li> | ||||
| </ul> | ||||
|  | ||||
| <h2> | ||||
|   Clubs : | ||||
| </h2> | ||||
| <ul> | ||||
|   <li>Positifs : {{ summary.total_positive_club }} soit {{ summary.balance_positive_club / 100 }} €</li> | ||||
|   <li>Neutres : {{ summary.total_zero_club }}</li> | ||||
|   <li>Négatifs : {{ summary.total_negative_club }} soit {{ summary.balance_negative_club / 100 }} €</li> | ||||
| </ul> | ||||
|  | ||||
| <h2> | ||||
|   Clubs hors BDE / Kfet et club dont le nom fini par "- BDE" : | ||||
| </h2> | ||||
| <ul> | ||||
|   <li>Positifs : {{ summary.total_positive_club_nbde }} soit {{ summary.balance_positive_club_nbde / 100 }} €</li> | ||||
|   <li>Neutres : {{ summary.total_zero_club_nbde }}</li> | ||||
|   <li>Négatifs : {{ summary.total_negative_club_nbde }} soit {{ summary.balance_negative_club_nbde / 100 }} €</li> | ||||
| </ul> | ||||
|  | ||||
| <h2> | ||||
|   Progression : | ||||
| </h2> | ||||
| <ul> | ||||
|   <li>Ceci correspond à une différence de {{ balance_difference_user / 100 }} € pour les utilisateur⋅rices</li> | ||||
|   <li>Ceci correspond à une différence de {{ balance_difference_club / 100 }} € pour les clubs</li> | ||||
| </ul> | ||||
|  | ||||
| -- | ||||
| <p> | ||||
|     Le BDE<br> | ||||
|     {% trans "Mail generated by the Note Kfet on the" %} {% now "j F Y à H:i:s" %} | ||||
| </p> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										33
									
								
								apps/note/templates/note/mails/summary_notes_report.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								apps/note/templates/note/mails/summary_notes_report.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| {% load pretty_money %} | ||||
| {% load i18n %} | ||||
|  | ||||
| Récapitulatif de trésorerie au {{ summary.date|date:"d/m/Y" }} à {{ summary.date|date:"H:i:s" }} : | ||||
|  | ||||
| Tous les utilisateur⋅rices : | ||||
|   - Positifs : {{ summary.total_positive_user }} soit {{ summary.balance_positive_user / 100 }} € | ||||
|   - Neutres : {{ summary.total_zero_user }} | ||||
|   - Négatifs : {{ summary.total_negative_user }} soit {{ summary.balance_negative_user / 100 }} € | ||||
|  | ||||
| Les {{ summary.total_positive_user_bde + summary.total_zero_user_bde + summary.total_negative_user_bde }} adhérent⋅es BDE : | ||||
|   - Positifs : {{ summary.total_positive_user_bde }} soit {{ summary.balance_positive_user_bde / 100 }} € | ||||
|   - Neutres : {{ summary.total_zero_user_bde }} | ||||
|   - Négatifs : {{ summary.total_negative_user_bde }} soit {{ summary.balance_negative_user_bde /100 }} € | ||||
|  | ||||
| Clubs : | ||||
|   - Positifs : {{ summary.total_positive_club }} soit {{ summary.balance_positive_club / 100 }} € | ||||
|   - Neutres : {{ summary.total_zero_club }} | ||||
|   - Négatifs : {{ summary.total_negative_club }} soit {{ summary.balance_negative_club / 100 }} € | ||||
|  | ||||
| Clubs hors BDE / Kfet et club dont le nom fini par "- BDE" : | ||||
|   - Positifs : {{ summary.total_positive_club_nbde }} soit {{ summary.balance_positive_club_nbde / 100 }} € | ||||
|   - Neutres : {{ summary.total_zero_club_nbde }} | ||||
|   - Négatifs : {{ summary.total_negative_club_nbde }} soit {{ summary.balance_negative_club_nbde / 100 }} € | ||||
|  | ||||
| Progression : | ||||
|   - Ceci correspond à une différence de {{ balance_difference_user / 100 }} € pour les utilisateur⋅rices | ||||
|   - Ceci correspond à une différence de {{ balance_difference_club / 100 }} € pour les clubs | ||||
|  | ||||
| -- | ||||
| Le BDE | ||||
|  | ||||
| {% trans "Mail generated by the Note Kfet on the" %} {% now "j F Y à H:i:s" %} | ||||
| @@ -2591,12 +2591,12 @@ | ||||
| 				"note", | ||||
| 				"transaction" | ||||
| 			], | ||||
| 			"query": "[\"OR\", {\"source__balance__gte\": {\"F\": [\"SUB\", [\"MUL\", [\"F\", \"amount\"], [\"F\", \"quantity\"]], 2000]}}, {\"valid\": false}]", | ||||
| 			"query": "[\"OR\", {\"source__balance__gte\": 0}, [\"AND\", [\"NOT\", {\"recurrenttransaction__template__category__name\": \"Alcool\"}], {\"source__balance__gte\": {\"F\": [\"SUB\", [\"MUL\", [\"F\", \"amount\"], [\"F\", \"quantity\"]], 2000]}}], {\"valid\": false}]", | ||||
| 			"type": "add", | ||||
| 			"mask": 2, | ||||
| 			"field": "", | ||||
| 			"permanent": false, | ||||
| 			"description": "Créer une transaction quelconque tant que la source reste au-dessus de -20 €" | ||||
| 			"description": "Créer une transaction quelconque tant que la source reste positive s'il s'agit d'alcool, sinon au-dessus de -20€" | ||||
| 		} | ||||
| 	}, | ||||
| 	{ | ||||
|   | ||||
| @@ -5,7 +5,6 @@ from django import forms | ||||
| from django.contrib.auth.forms import UserCreationForm | ||||
| from django.contrib.auth.models import User | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| # from member.models import Club | ||||
| from note.models import NoteSpecial, Alias | ||||
| from note_kfet.inputs import AmountInput | ||||
|  | ||||
| @@ -115,12 +114,3 @@ class ValidationForm(forms.Form): | ||||
|         required=False, | ||||
|         initial=True, | ||||
|     ) | ||||
|  | ||||
| #     If the bda exists | ||||
| #     if Club.objects.filter(name__iexact="bda").exists(): | ||||
| #         The user can join the bda club at the inscription | ||||
| #         join_bda = forms.BooleanField( | ||||
| #             label=_("Join BDA Club"), | ||||
| #             required=False, | ||||
| #             initial=True, | ||||
| #         ) | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from django import forms | ||||
| from django.conf import settings | ||||
| from django.contrib.auth.mixins import LoginRequiredMixin | ||||
| from django.contrib.auth.models import User | ||||
| @@ -238,9 +239,8 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|         fee += bde.membership_fee_paid if user.profile.paid else bde.membership_fee_unpaid | ||||
|         kfet = Club.objects.get(name="Kfet") | ||||
|         fee += kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid | ||||
|         if Club.objects.filter(name__iexact="BDA").exists(): | ||||
|             bda = Club.objects.get(name__iexact="BDA") | ||||
|             fee += bda.membership_fee_paid if user.profile.paid else bda.membership_fee_unpaid | ||||
|         for club in Club.objects.filter(add_registration_form=True): | ||||
|             fee += club.membership_fee_paid if user.profile.paid else club.membership_fee_unpaid | ||||
|         ctx["total_fee"] = "{:.02f}".format(fee / 100, ) | ||||
|  | ||||
| #        ctx["declare_soge_account"] = SogeCredit.objects.filter(user=user).exists() | ||||
| @@ -249,6 +249,16 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|  | ||||
|     def get_form(self, form_class=None): | ||||
|         form = super().get_form(form_class) | ||||
|  | ||||
|         # add clubs that are in registration form | ||||
|         for club in Club.objects.filter(add_registration_form=True).order_by("name"): | ||||
|             form_join_club = forms.BooleanField( | ||||
|                 label=_("Join %(club)s Club") % {'club': club.name}, | ||||
|                 required=False, | ||||
|                 initial=False, | ||||
|             ) | ||||
|             form.fields.update({f"join_{club.id}": form_join_club}) | ||||
|  | ||||
|         user = self.get_object() | ||||
|         form.fields["last_name"].initial = user.last_name | ||||
|         form.fields["first_name"].initial = user.first_name | ||||
| @@ -266,11 +276,6 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|             form.add_error(None, _("An alias with a similar name already exists.")) | ||||
|             return self.form_invalid(form) | ||||
|  | ||||
|         # Check if BDA exist to propose membership at regisration | ||||
|         bda_exists = False | ||||
|         if Club.objects.filter(name__iexact="BDA").exists(): | ||||
|             bda_exists = True | ||||
|  | ||||
|         # Get form data | ||||
| #        soge = form.cleaned_data["soge"] | ||||
|         credit_type = form.cleaned_data["credit_type"] | ||||
| @@ -280,8 +285,9 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|         bank = form.cleaned_data["bank"] | ||||
|         join_bde = form.cleaned_data["join_bde"] | ||||
|         join_kfet = form.cleaned_data["join_kfet"] | ||||
|         if bda_exists: | ||||
|             join_bda = form.cleaned_data["join_bda"] | ||||
|  | ||||
|         clubs_registration = Club.objects.filter(add_registration_form=True).order_by("name") | ||||
|         join_clubs = [(club, form.cleaned_data[f"join_{club.id}"]) for club in clubs_registration] | ||||
|  | ||||
| #        if soge: | ||||
| #            # If Société Générale pays the inscription, the user automatically joins the two clubs. | ||||
| @@ -303,11 +309,12 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|         kfet_fee = kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid | ||||
|         # Add extra fee for the full membership | ||||
|         fee += kfet_fee if join_kfet else 0 | ||||
|         if bda_exists: | ||||
|             bda = Club.objects.get(name__iexact="BDA") | ||||
|             bda_fee = bda.membership_fee_paid if user.profile.paid else bda.membership_fee_unpaid | ||||
|             # Add extra fee for the bda membership | ||||
|             fee += bda_fee if join_bda else 0 | ||||
|         clubs_fee = dict() | ||||
|         for club, join_club in join_clubs: | ||||
|             club_fee = club.membership_fee_paid if user.profile.paid else club.membership_fee_unpaid | ||||
|             # Add extra fee for the club membership | ||||
|             clubs_fee[club] = club_fee | ||||
|             fee += club_fee if join_club else 0 | ||||
|  | ||||
| #        # If the bank pays, then we don't credit now. Treasurers will validate the transaction | ||||
| #        # and credit the note later. | ||||
| @@ -387,17 +394,18 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | ||||
|             membership.roles.add(Role.objects.get(name="Adhérent Kfet")) | ||||
|             membership.save() | ||||
|  | ||||
|         if bda_exists and join_bda: | ||||
|             # Create membership for the user to the BDA starting today | ||||
|             membership = Membership( | ||||
|                 club=bda, | ||||
|                 user=user, | ||||
|                 fee=bda_fee, | ||||
|             ) | ||||
|             membership.save() | ||||
|             membership.refresh_from_db() | ||||
|             membership.roles.add(Role.objects.get(name="Membre de club")) | ||||
|             membership.save() | ||||
|         for club, join_club in join_clubs: | ||||
|             if join_club: | ||||
|                 # Create membership for the user to the BDA starting today | ||||
|                 membership = Membership( | ||||
|                     club=club, | ||||
|                     user=user, | ||||
|                     fee=clubs_fee[club], | ||||
|                 ) | ||||
|                 membership.save() | ||||
|                 membership.refresh_from_db() | ||||
|                 membership.roles.add(Role.objects.get(name="Membre de club")) | ||||
|                 membership.save() | ||||
|  | ||||
| #        if soge: | ||||
| #            soge_credit = SogeCredit.objects.get(user=user) | ||||
|   | ||||
 Submodule apps/scripts updated: f580f9b9e9...f76acb3248
									
								
							| @@ -5,13 +5,13 @@ from django.contrib import admin | ||||
| from note_kfet.admin import admin_site | ||||
|  | ||||
| from .forms import ProductForm | ||||
| from .models import RemittanceType, Remittance, SogeCredit, Invoice, Product | ||||
| from .models import Invoice, NoteSummary, Product, RemittanceType, Remittance, SogeCredit | ||||
|  | ||||
|  | ||||
| @admin.register(RemittanceType, site=admin_site) | ||||
| class RemittanceTypeAdmin(admin.ModelAdmin): | ||||
|     """ | ||||
|     Admin customisation for RemiitanceType | ||||
|     Admin customisation for RemittanceType | ||||
|     """ | ||||
|     list_display = ('note', ) | ||||
|  | ||||
| @@ -55,3 +55,19 @@ class InvoiceAdmin(admin.ModelAdmin): | ||||
|     """ | ||||
|     list_display = ('object', 'id', 'bde', 'name', 'date', 'acquitted',) | ||||
|     inlines = (ProductInline,) | ||||
|  | ||||
|  | ||||
| @admin.register(NoteSummary, site=admin_site) | ||||
| class NoteSummaryAdmin(admin.ModelAdmin): | ||||
|     """ | ||||
|     Admin customisation for NoteSummary | ||||
|     """ | ||||
|     list_display = ( | ||||
|         'date', 'total_positive_user', 'balance_positive_user', 'total_positive_user_bde', | ||||
|         'balance_positive_user_bde', 'total_zero_user', 'total_zero_user_bde', 'total_negative_user', | ||||
|         'balance_negative_user', 'total_negative_user_bde', 'balance_negative_user_bde', | ||||
|         'total_vnegative_user', 'balance_vnegative_user', 'total_vnegative_user_bde', | ||||
|         'balance_vnegative_user_bde', 'total_positive_club', 'balance_positive_club', | ||||
|         'total_positive_club_nbde', 'balance_positive_club_nbde', 'total_zero_club', 'total_zero_club_nbde', | ||||
|         'total_negative_club', 'balance_negative_club', 'total_negative_club_nbde', 'balance_negative_club_nbde', | ||||
|     ) | ||||
|   | ||||
							
								
								
									
										49
									
								
								apps/treasury/migrations/0009_notesummary.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								apps/treasury/migrations/0009_notesummary.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| # Generated by Django 2.2.28 on 2024-08-07 12:09 | ||||
|  | ||||
| import datetime | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ('treasury', '0008_auto_20240322_0045'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.CreateModel( | ||||
|             name='NoteSummary', | ||||
|             fields=[ | ||||
|                 ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), | ||||
|                 ('date', models.DateField(default=datetime.date.today, verbose_name='Date')), | ||||
|                 ('total_positive_user', models.PositiveIntegerField(verbose_name='Total positive user')), | ||||
|                 ('balance_positive_user', models.PositiveIntegerField(verbose_name='Balance positive user')), | ||||
|                 ('total_positive_user_bde', models.PositiveIntegerField(verbose_name='Total positive user BDE')), | ||||
|                 ('balance_positive_user_bde', models.PositiveIntegerField(verbose_name='Balance positive user BDE')), | ||||
|                 ('total_zero_user', models.PositiveIntegerField(verbose_name='Total zero user')), | ||||
|                 ('total_zero_user_bde', models.PositiveIntegerField(verbose_name='Total zero user BDE')), | ||||
|                 ('total_negative_user', models.PositiveIntegerField(verbose_name='Total negative user')), | ||||
|                 ('balance_negative_user', models.PositiveIntegerField(verbose_name='Balance negative user')), | ||||
|                 ('total_negative_user_bde', models.PositiveIntegerField(verbose_name='Total negative user BDE')), | ||||
|                 ('balance_negative_user_bde', models.PositiveIntegerField(verbose_name='Balance negative user BDE')), | ||||
|                 ('total_vnegative_user', models.PositiveIntegerField(verbose_name='Total very negative user')), | ||||
|                 ('balance_vnegative_user', models.PositiveIntegerField(verbose_name='Balance very negative user')), | ||||
|                 ('total_vnegative_user_bde', models.PositiveIntegerField(verbose_name='Total very negative user BDE')), | ||||
|                 ('balance_vnegative_user_bde', models.PositiveIntegerField(verbose_name='Balance very negative user BDE')), | ||||
|                 ('total_positive_club', models.PositiveIntegerField(verbose_name='Total positive club')), | ||||
|                 ('balance_positive_club', models.PositiveIntegerField(verbose_name='Balance positive club')), | ||||
|                 ('total_positive_club_nbde', models.PositiveIntegerField(verbose_name='Total positive club nbde')), | ||||
|                 ('balance_positive_club_nbde', models.PositiveIntegerField(verbose_name='Balance positive club nbde')), | ||||
|                 ('total_zero_club', models.PositiveIntegerField(verbose_name='Total zero club')), | ||||
|                 ('total_zero_club_nbde', models.PositiveIntegerField(verbose_name='Total zero club nbde')), | ||||
|                 ('total_negative_club', models.PositiveIntegerField(verbose_name='Total negative club')), | ||||
|                 ('balance_negative_club', models.PositiveIntegerField(verbose_name='Balance negative club')), | ||||
|                 ('total_negative_club_nbde', models.PositiveIntegerField(verbose_name='Total negative club nbde')), | ||||
|                 ('balance_negative_club_nbde', models.PositiveIntegerField(verbose_name='Balance negative club nbde')), | ||||
|             ], | ||||
|             options={ | ||||
|                 'verbose_name': 'Summary', | ||||
|                 'verbose_name_plural': 'Summaries', | ||||
|             }, | ||||
|         ), | ||||
|     ] | ||||
| @@ -460,3 +460,117 @@ class SogeCredit(models.Model): | ||||
|             self.credit_transaction._force_save = True | ||||
|             self.credit_transaction.save() | ||||
|         super().delete(**kwargs) | ||||
|  | ||||
|  | ||||
| class NoteSummary(models.Model): | ||||
|     """ | ||||
|     Summary of every notes | ||||
|     """ | ||||
|  | ||||
|     date = models.DateField( | ||||
|         default=date.today, | ||||
|         verbose_name=_("Date"), | ||||
|     ) | ||||
|  | ||||
|     total_positive_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total positive user"), | ||||
|     ) | ||||
|  | ||||
|     balance_positive_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance positive user"), | ||||
|     ) | ||||
|  | ||||
|     total_positive_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total positive user BDE"), | ||||
|     ) | ||||
|  | ||||
|     balance_positive_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance positive user BDE"), | ||||
|     ) | ||||
|  | ||||
|     total_zero_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total zero user"), | ||||
|     ) | ||||
|  | ||||
|     total_zero_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total zero user BDE"), | ||||
|     ) | ||||
|  | ||||
|     total_negative_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total negative user"), | ||||
|     ) | ||||
|  | ||||
|     balance_negative_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance negative user"), | ||||
|     ) | ||||
|  | ||||
|     total_negative_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total negative user BDE"), | ||||
|     ) | ||||
|  | ||||
|     balance_negative_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance negative user BDE"), | ||||
|     ) | ||||
|  | ||||
|     total_vnegative_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total very negative user"), | ||||
|     ) | ||||
|  | ||||
|     balance_vnegative_user = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance very negative user"), | ||||
|     ) | ||||
|  | ||||
|     total_vnegative_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total very negative user BDE"), | ||||
|     ) | ||||
|  | ||||
|     balance_vnegative_user_bde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance very negative user BDE"), | ||||
|     ) | ||||
|  | ||||
|     total_positive_club = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total positive club"), | ||||
|     ) | ||||
|  | ||||
|     balance_positive_club = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance positive club"), | ||||
|     ) | ||||
|  | ||||
|     total_positive_club_nbde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total positive club nbde"), | ||||
|     ) | ||||
|  | ||||
|     balance_positive_club_nbde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance positive club nbde"), | ||||
|     ) | ||||
|  | ||||
|     total_zero_club = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total zero club"), | ||||
|     ) | ||||
|  | ||||
|     total_zero_club_nbde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total zero club nbde"), | ||||
|     ) | ||||
|  | ||||
|     total_negative_club = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total negative club"), | ||||
|     ) | ||||
|  | ||||
|     balance_negative_club = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance negative club"), | ||||
|     ) | ||||
|  | ||||
|     total_negative_club_nbde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Total negative club nbde"), | ||||
|     ) | ||||
|  | ||||
|     balance_negative_club_nbde = models.PositiveIntegerField( | ||||
|         verbose_name=_("Balance negative club nbde"), | ||||
|     ) | ||||
|  | ||||
|     class Meta: | ||||
|         verbose_name = _("Summary") | ||||
|         verbose_name_plural = _("Summaries") | ||||
|  | ||||
|     def __str__(self): | ||||
|         return "Note summary of {date}".format(date=self.date) | ||||
|   | ||||
| @@ -2,11 +2,11 @@ | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm | ||||
| from .wei2024 import WEISurvey2024 | ||||
| from .wei2023 import WEISurvey2023 | ||||
|  | ||||
|  | ||||
| __all__ = [ | ||||
|     'WEISurvey', 'WEISurveyInformation', 'WEISurveyAlgorithm', 'CurrentSurvey', | ||||
| ] | ||||
|  | ||||
| CurrentSurvey = WEISurvey2024 | ||||
| CurrentSurvey = WEISurvey2023 | ||||
|   | ||||
| @@ -1,337 +0,0 @@ | ||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from functools import lru_cache | ||||
|  | ||||
| from django import forms | ||||
| from django.db import transaction | ||||
| from django.db.models import Q | ||||
|  | ||||
| from .base import WEISurvey, WEISurveyInformation, WEISurveyAlgorithm, WEIBusInformation | ||||
| from ...models import WEIMembership | ||||
|  | ||||
|  | ||||
| buses_descr = [ | ||||
|     [ | ||||
|         "Magi[Kar]p", "#ef5568", | ||||
|         """bus faible en alcool mais fort en connerie avec une partie calme pour les amateurs de sieste et de jeux de société. | ||||
|         Non discriminant il accepte tout le monde y compris le plus nulle des pokémons (magicarpe !!!!!!). Malgré les | ||||
|         accusations mensongères, il n'y a aucun weeb dans le Magi[Kar]p""", | ||||
|     ], | ||||
|     [ | ||||
|         "Va[car]me", "#fd7a28", | ||||
|         "descr", | ||||
|     ], | ||||
|     [ | ||||
|         "[Kar]aïbes", "#a5cfdd", | ||||
|         "descr", | ||||
|     ], | ||||
|     [ | ||||
|         "[Kar]di [Bus]", "#e46398", | ||||
|         "descr", | ||||
|     ], | ||||
|     [ | ||||
|         "Sparta[bus] 🐺🐒🏉", "#ebdac2", | ||||
|         "descr", | ||||
|     ], | ||||
|     [ | ||||
|         "Zanzo[Bus]", "#FFFF", | ||||
|         "descr", | ||||
|     ], | ||||
|     [ | ||||
|         "Bran[Kar]", "#6da1ac", | ||||
|         """Si vous ne connaissez pas le Bran[Kar], c’est comme une grande famille qui fait un apéro, qui se bourre un peu la | ||||
|         gueule en discutant des heures autour d’une table remplie de bouffe et de super bons cocktails (la plupart des | ||||
|         barmen/barwomen du bus sont les barmans de Shakens), sauf qu’on est un bus du Wei (vous comprendrez bien le nom de notre | ||||
|         bus en voyant l’état de certain·e·s). Il nous arrive de faire quelques conneries, mais surtout de jouer au Bière-pong en | ||||
|         musique !""", | ||||
|     ], | ||||
|     [ | ||||
|         "Techno [kar]ade", "#8065a3", | ||||
|         "descr" | ||||
|     ], | ||||
|     [ | ||||
|         "[Bus]ka-P", "#7c4768", | ||||
|         "descr", | ||||
|     ], | ||||
| ] | ||||
|  | ||||
|  | ||||
| def get_survey_info(id): | ||||
|     s = {"recap": { | ||||
|         "1": 0, | ||||
|         "2": 0, | ||||
|         "3": 0, | ||||
|         "4": 0, | ||||
|         "5": 0, | ||||
|     }} | ||||
|     s_ = {f"bus{id}": {f"{i}": 0 for i in range(1, 5 + 1)} for id in range(len(buses_descr))} | ||||
|     s.update(s_) | ||||
|     s.update({f"bus{id}": {f"{i}": i for i in range(1, 5 + 1)}}) | ||||
|     return {"scores": s} | ||||
|  | ||||
|  | ||||
| def print_bus(id): | ||||
|     return buses_descr[id][0] + "\n\n" + buses_descr[id][2] | ||||
|  | ||||
|  | ||||
| def print_all_buses(): | ||||
|     liste = [print_bus(id) for id in range(len(buses_descr))] | ||||
|     return "<br><br>---------<br><br>".join(liste) | ||||
|  | ||||
|  | ||||
| WORDS = { | ||||
|     "recap": | ||||
|         [ | ||||
|             """Chèr⋅e 1A, te voilà arrivé⋅e devant un choix fatidique, le choix de ton bus.......<br> | ||||
|             (Musique effrayante)<br> | ||||
|             Petite blagounette évidemment, chacun des bus te permettra de passer un excellent WEI ! | ||||
|             Mais quitte à avoir le choix, voici la liste de tous les bus ainsi qu'une description détaillée de ces derniers ! | ||||
|             Prends ton temps, observe les bien et quand tu te sens prêt⋅e, appuye sur le bouton 'Noter les bus' pour continuer | ||||
|             (pas besoin d'apprendre par cœur les bus, la description du bus te sera rappeler avant de le noter !) <br><br><br>""" + print_all_buses(), | ||||
|             { | ||||
|                 1: "Noter les bus :", | ||||
|             } | ||||
|         ] | ||||
| } | ||||
|  | ||||
| WORDS.update({ | ||||
|     f"bus{id}": [print_bus(id), {i: f"Noter {i}/5" for i in range(1, 5 + 1)}] for id in range(len(buses_descr)) | ||||
| }) | ||||
|  | ||||
|  | ||||
| class WEISurveyForm2024(forms.Form): | ||||
|     """ | ||||
|     Survey form for the year 2024. | ||||
|     Members answer 20 questions, from which we calculate the best associated bus. | ||||
|     """ | ||||
|     def set_registration(self, registration): | ||||
|         """ | ||||
|         Filter the bus selector with the buses of the current WEI. | ||||
|         """ | ||||
|         information = WEISurveyInformation2024(registration) | ||||
|  | ||||
|         question = information.questions[information.step] | ||||
|         self.fields[question] = forms.ChoiceField( | ||||
|             label=WORDS[question][0], | ||||
|             widget=forms.RadioSelect(), | ||||
|         ) | ||||
|         answers = [(answer, WORDS[question][1][answer]) for answer in WORDS[question][1]] | ||||
|         self.fields[question].choices = answers | ||||
|  | ||||
|  | ||||
| class WEIBusInformation2024(WEIBusInformation): | ||||
|     """ | ||||
|     For each question, the bus has ordered answers | ||||
|     """ | ||||
|     scores: dict | ||||
|  | ||||
|     def __init__(self, bus): | ||||
|         self.scores = {} | ||||
|         for question in WORDS: | ||||
|             self.scores[question] = [] | ||||
|         super().__init__(bus) | ||||
|  | ||||
|  | ||||
| class WEISurveyInformation2024(WEISurveyInformation): | ||||
|     """ | ||||
|     We store the id of the selected bus. We store only the name, but is not used in the selection: | ||||
|     that's only for humans that try to read data. | ||||
|     """ | ||||
|  | ||||
|     step = 0 | ||||
|     questions = list(WORDS.keys()) | ||||
|  | ||||
|     def __init__(self, registration): | ||||
|         for question in WORDS: | ||||
|             setattr(self, str(question), None) | ||||
|         super().__init__(registration) | ||||
|  | ||||
|  | ||||
| class WEISurvey2024(WEISurvey): | ||||
|     """ | ||||
|     Survey for the year 2024. | ||||
|     """ | ||||
|  | ||||
|     @classmethod | ||||
|     def get_year(cls): | ||||
|         return 2024 | ||||
|  | ||||
|     @classmethod | ||||
|     def get_survey_information_class(cls): | ||||
|         return WEISurveyInformation2024 | ||||
|  | ||||
|     def get_form_class(self): | ||||
|         return WEISurveyForm2024 | ||||
|  | ||||
|     def update_form(self, form): | ||||
|         """ | ||||
|         Filter the bus selector with the buses of the WEI. | ||||
|         """ | ||||
|         form.set_registration(self.registration) | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         self.information.step += 1 | ||||
|         for question in WORDS: | ||||
|             if question in form.cleaned_data: | ||||
|                 answer = form.cleaned_data[question] | ||||
|                 setattr(self.information, question, answer) | ||||
|         self.save() | ||||
|  | ||||
|     @classmethod | ||||
|     def get_algorithm_class(cls): | ||||
|         return WEISurveyAlgorithm2024 | ||||
|  | ||||
|     def is_complete(self) -> bool: | ||||
|         """ | ||||
|         The survey is complete once the bus is chosen. | ||||
|         """ | ||||
|         for question in WORDS: | ||||
|             if not getattr(self.information, question): | ||||
|                 return False | ||||
|         return True | ||||
|  | ||||
|     @lru_cache() | ||||
|     def score(self, bus): | ||||
|         if not self.is_complete(): | ||||
|             raise ValueError("Survey is not ended, can't calculate score") | ||||
|  | ||||
|         bus_info = self.get_algorithm_class().get_bus_information(bus) | ||||
|         # Score is the given score by the bus subtracted to the mid-score of the buses. | ||||
|         s = 0 | ||||
|         for question in WORDS: | ||||
|             s += bus_info.scores[question][str(getattr(self.information, question))] | ||||
|         return s | ||||
|  | ||||
|     @lru_cache() | ||||
|     def scores_per_bus(self): | ||||
|         return {bus: self.score(bus) for bus in self.get_algorithm_class().get_buses()} | ||||
|  | ||||
|     @lru_cache() | ||||
|     def ordered_buses(self): | ||||
|         values = list(self.scores_per_bus().items()) | ||||
|         values.sort(key=lambda item: -item[1]) | ||||
|         return values | ||||
|  | ||||
|     @classmethod | ||||
|     def clear_cache(cls): | ||||
|         return super().clear_cache() | ||||
|  | ||||
|  | ||||
| class WEISurveyAlgorithm2024(WEISurveyAlgorithm): | ||||
|     """ | ||||
|     The algorithm class for the year 2024. | ||||
|     We use Gale-Shapley algorithm to attribute 1y students into buses. | ||||
|     """ | ||||
|  | ||||
|     @classmethod | ||||
|     def get_survey_class(cls): | ||||
|         return WEISurvey2024 | ||||
|  | ||||
|     @classmethod | ||||
|     def get_bus_information_class(cls): | ||||
|         return WEIBusInformation2024 | ||||
|  | ||||
|     def run_algorithm(self, display_tqdm=False): | ||||
|         """ | ||||
|         Gale-Shapley algorithm implementation. | ||||
|         We modify it to allow buses to have multiple "weddings". | ||||
|         """ | ||||
|         surveys = list(self.get_survey_class()(r) for r in self.get_registrations())  # All surveys | ||||
|         surveys = [s for s in surveys if s.is_complete()]  # Don't consider invalid surveys | ||||
|         # Don't manage hardcoded people | ||||
|         surveys = [s for s in surveys if not hasattr(s.information, 'hardcoded') or not s.information.hardcoded] | ||||
|  | ||||
|         # Reset previous algorithm run | ||||
|         for survey in surveys: | ||||
|             survey.free() | ||||
|             survey.save() | ||||
|  | ||||
|         non_men = [s for s in surveys if s.registration.gender != 'male'] | ||||
|         men = [s for s in surveys if s.registration.gender == 'male'] | ||||
|  | ||||
|         quotas = {} | ||||
|         registrations = self.get_registrations() | ||||
|         non_men_total = registrations.filter(~Q(gender='male')).count() | ||||
|         for bus in self.get_buses(): | ||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() | ||||
|             # Remove hardcoded people | ||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, | ||||
|                                                        registration__information_json__icontains="hardcoded").count() | ||||
|             quotas[bus] = 4 + int(non_men_total / registrations.count() * free_seats) | ||||
|  | ||||
|         tqdm_obj = None | ||||
|         if display_tqdm: | ||||
|             from tqdm import tqdm | ||||
|             tqdm_obj = tqdm(total=len(non_men), desc="Non-hommes") | ||||
|  | ||||
|         # Repartition for non men people first | ||||
|         self.make_repartition(non_men, quotas, tqdm_obj=tqdm_obj) | ||||
|  | ||||
|         quotas = {} | ||||
|         for bus in self.get_buses(): | ||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() | ||||
|             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) | ||||
|             # Remove hardcoded people | ||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, | ||||
|                                                        registration__information_json__icontains="hardcoded").count() | ||||
|             quotas[bus] = free_seats | ||||
|  | ||||
|         if display_tqdm: | ||||
|             tqdm_obj.close() | ||||
|  | ||||
|             from tqdm import tqdm | ||||
|             tqdm_obj = tqdm(total=len(men), desc="Hommes") | ||||
|  | ||||
|         self.make_repartition(men, quotas, tqdm_obj=tqdm_obj) | ||||
|  | ||||
|         if display_tqdm: | ||||
|             tqdm_obj.close() | ||||
|  | ||||
|         # Clear cache information after running algorithm | ||||
|         WEISurvey2024.clear_cache() | ||||
|  | ||||
|     def make_repartition(self, surveys, quotas=None, tqdm_obj=None): | ||||
|         free_surveys = surveys.copy()  # Remaining surveys | ||||
|         while free_surveys:  # Some students are not affected | ||||
|             survey = free_surveys[0] | ||||
|             buses = survey.ordered_buses()  # Preferences of the student | ||||
|             for bus, current_score in buses: | ||||
|                 if self.get_bus_information(bus).has_free_seats(surveys, quotas): | ||||
|                     # Selected bus has free places. Put student in the bus | ||||
|                     survey.select_bus(bus) | ||||
|                     survey.save() | ||||
|                     free_surveys.remove(survey) | ||||
|                     break | ||||
|                 else: | ||||
|                     # Current bus has not enough places. Remove the least preferred student from the bus if existing | ||||
|                     least_preferred_survey = None | ||||
|                     least_score = -1 | ||||
|                     # Find the least student in the bus that has a lower score than the current student | ||||
|                     for survey2 in surveys: | ||||
|                         if not survey2.information.valid or survey2.information.get_selected_bus() != bus: | ||||
|                             continue | ||||
|                         score2 = survey2.score(bus) | ||||
|                         if current_score <= score2:  # Ignore better students | ||||
|                             continue | ||||
|                         if least_preferred_survey is None or score2 < least_score: | ||||
|                             least_preferred_survey = survey2 | ||||
|                             least_score = score2 | ||||
|  | ||||
|                     if least_preferred_survey is not None: | ||||
|                         # Remove the least student from the bus and put the current student in. | ||||
|                         # If it does not exist, choose the next bus. | ||||
|                         least_preferred_survey.free() | ||||
|                         least_preferred_survey.save() | ||||
|                         free_surveys.append(least_preferred_survey) | ||||
|                         survey.select_bus(bus) | ||||
|                         survey.save() | ||||
|                         free_surveys.remove(survey) | ||||
|                         break | ||||
|             else: | ||||
|                 raise ValueError(f"User {survey.registration.user} has no free seat") | ||||
|  | ||||
|             if tqdm_obj is not None: | ||||
|                 tqdm_obj.n = len(surveys) - len(free_surveys) | ||||
|                 tqdm_obj.refresh() | ||||
| @@ -6,6 +6,8 @@ from datetime import date, timedelta | ||||
|  | ||||
| from django.contrib.auth.models import User | ||||
| from django.test import TestCase | ||||
| from django.urls import reverse | ||||
| from note.models import NoteUser | ||||
|  | ||||
| from ..forms.surveys.wei2023 import WEIBusInformation2023, WEISurvey2023, WORDS, WEISurveyInformation2023 | ||||
| from ..models import Bus, WEIClub, WEIRegistration | ||||
| @@ -125,3 +127,44 @@ class TestWEIAlgorithm(TestCase): | ||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance | ||||
|  | ||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % | ||||
|  | ||||
|     def test_register_1a(self): | ||||
|         """ | ||||
|         Test register a first year member to the WEI and complete the survey | ||||
|         """ | ||||
|         response = self.client.get(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         user = User.objects.create(username="toto", email="toto@example.com") | ||||
|         NoteUser.objects.create(user=user) | ||||
|         response = self.client.post(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk)), dict( | ||||
|             user=user.id, | ||||
|             soge_credit=True, | ||||
|             birth_date=date(2000, 1, 1), | ||||
|             gender='nonbinary', | ||||
|             clothing_cut='female', | ||||
|             clothing_size='XS', | ||||
|             health_issues='I am a bot', | ||||
|             emergency_contact_name='NoteKfet2020', | ||||
|             emergency_contact_phone='+33123456789', | ||||
|         )) | ||||
|         qs = WEIRegistration.objects.filter(user_id=user.id) | ||||
|         self.assertTrue(qs.exists()) | ||||
|         registration = qs.get() | ||||
|         self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, 200) | ||||
|         for question in WORDS: | ||||
|             # Fill 1A Survey, 20 pages | ||||
|             # be careful if questionnary form change (number of page, type of answer...) | ||||
|             response = self.client.post(reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), { | ||||
|                 question: "1" | ||||
|             }) | ||||
|             registration.refresh_from_db() | ||||
|             survey = WEISurvey2023(registration) | ||||
|             self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, | ||||
|                                  302 if survey.is_complete() else 200) | ||||
|             self.assertIsNotNone(getattr(survey.information, question), "Survey page " + question + " failed") | ||||
|         survey = WEISurvey2023(registration) | ||||
|         self.assertTrue(survey.is_complete()) | ||||
|         survey.select_bus(self.buses[0]) | ||||
|         survey.save() | ||||
|         self.assertIsNotNone(survey.information.get_selected_bus()) | ||||
|   | ||||
| @@ -1,172 +0,0 @@ | ||||
| # Copyright (C) 2018-2024 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| import random | ||||
| from datetime import date, timedelta | ||||
|  | ||||
| from django.contrib.auth.models import User | ||||
| from django.test import TestCase | ||||
| from django.urls import reverse | ||||
| from note.models import NoteUser | ||||
|  | ||||
| from ..forms.surveys.wei2024 import WEIBusInformation2024, WEISurvey2024, WORDS, WEISurveyInformation2024 | ||||
| from ..models import Bus, WEIClub, WEIRegistration | ||||
|  | ||||
|  | ||||
| class TestWEIAlgorithm(TestCase): | ||||
|     """ | ||||
|     Run some tests to ensure that the WEI algorithm is working well. | ||||
|     """ | ||||
|     fixtures = ('initial',) | ||||
|  | ||||
|     def setUp(self): | ||||
|         """ | ||||
|         Create some test data, with one WEI and 10 buses with random score attributions. | ||||
|         """ | ||||
|         self.user = User.objects.create_superuser( | ||||
|             username="weiadmin", | ||||
|             password="admin", | ||||
|             email="admin@example.com", | ||||
|         ) | ||||
|         self.user.save() | ||||
|         self.client.force_login(self.user) | ||||
|         sess = self.client.session | ||||
|         sess["permission_mask"] = 42 | ||||
|         sess.save() | ||||
|  | ||||
|         self.wei = WEIClub.objects.create( | ||||
|             name="WEI 2024", | ||||
|             email="wei2024@example.com", | ||||
|             parent_club_id=2, | ||||
|             membership_fee_paid=12500, | ||||
|             membership_fee_unpaid=5500, | ||||
|             membership_start='2024-01-01', | ||||
|             membership_end='2024-12-31', | ||||
|             date_start=date.today() + timedelta(days=2), | ||||
|             date_end='2024-12-31', | ||||
|             year=2024, | ||||
|         ) | ||||
|  | ||||
|         self.buses = [] | ||||
|         for i in range(10): | ||||
|             bus = Bus.objects.create(wei=self.wei, name=f"Bus {i}", size=10) | ||||
|             self.buses.append(bus) | ||||
|             information = WEIBusInformation2024(bus) | ||||
|             for question in WORDS: | ||||
|                 information.scores[question] = {answer: random.randint(1, 5) for answer in WORDS[question][1]} | ||||
|             information.save() | ||||
|             bus.save() | ||||
|  | ||||
|     def test_survey_algorithm_small(self): | ||||
|         """ | ||||
|         There are only a few people in each bus, ensure that each person has its best bus | ||||
|         """ | ||||
|         # Add a few users | ||||
|         for i in range(10): | ||||
|             user = User.objects.create(username=f"user{i}") | ||||
|             registration = WEIRegistration.objects.create( | ||||
|                 user=user, | ||||
|                 wei=self.wei, | ||||
|                 first_year=True, | ||||
|                 birth_date='2000-01-01', | ||||
|             ) | ||||
|             information = WEISurveyInformation2024(registration) | ||||
|             for question in WORDS: | ||||
|                 options = list(WORDS[question][1].keys()) | ||||
|                 setattr(information, question, random.choice(options)) | ||||
|             information.step = 20 | ||||
|             information.save(registration) | ||||
|             registration.save() | ||||
|  | ||||
|         # Run algorithm | ||||
|         WEISurvey2024.get_algorithm_class()().run_algorithm() | ||||
|  | ||||
|         # Ensure that everyone has its first choice | ||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): | ||||
|             survey = WEISurvey2024(r) | ||||
|             preferred_bus = survey.ordered_buses()[0][0] | ||||
|             chosen_bus = survey.information.get_selected_bus() | ||||
|             self.assertEqual(preferred_bus, chosen_bus) | ||||
|  | ||||
|     def test_survey_algorithm_full(self): | ||||
|         """ | ||||
|         Buses are full of first year people, ensure that they are happy | ||||
|         """ | ||||
|         # Add a lot of users | ||||
|         for i in range(95): | ||||
|             user = User.objects.create(username=f"user{i}") | ||||
|             registration = WEIRegistration.objects.create( | ||||
|                 user=user, | ||||
|                 wei=self.wei, | ||||
|                 first_year=True, | ||||
|                 birth_date='2000-01-01', | ||||
|             ) | ||||
|             information = WEISurveyInformation2024(registration) | ||||
|             for question in WORDS: | ||||
|                 options = list(WORDS[question][1].keys()) | ||||
|                 setattr(information, question, random.choice(options)) | ||||
|             information.step = 20 | ||||
|             information.save(registration) | ||||
|             registration.save() | ||||
|  | ||||
|         # Run algorithm | ||||
|         WEISurvey2024.get_algorithm_class()().run_algorithm() | ||||
|  | ||||
|         penalty = 0 | ||||
|         # Ensure that everyone seems to be happy | ||||
|         # We attribute a penalty for each user that didn't have its first choice | ||||
|         # The penalty is the square of the distance between the score of the preferred bus | ||||
|         # and the score of the attributed bus | ||||
|         # We consider it acceptable if the mean of this distance is lower than 5 % | ||||
|         for r in WEIRegistration.objects.filter(wei=self.wei).all(): | ||||
|             survey = WEISurvey2024(r) | ||||
|             chosen_bus = survey.information.get_selected_bus() | ||||
|             buses = survey.ordered_buses() | ||||
|             score = min(v for bus, v in buses if bus == chosen_bus) | ||||
|             max_score = buses[0][1] | ||||
|             penalty += (max_score - score) ** 2 | ||||
|  | ||||
|             self.assertLessEqual(max_score - score, 25)  # Always less than 25 % of tolerance | ||||
|  | ||||
|         self.assertLessEqual(penalty / 100, 25)  # Tolerance of 5 % | ||||
|  | ||||
|     def test_register_1a(self): | ||||
|         """ | ||||
|         Test register a first year member to the WEI and complete the survey | ||||
|         """ | ||||
|         response = self.client.get(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|         user = User.objects.create(username="toto", email="toto@example.com") | ||||
|         NoteUser.objects.create(user=user) | ||||
|         response = self.client.post(reverse("wei:wei_register_1A", kwargs=dict(wei_pk=self.wei.pk)), dict( | ||||
|             user=user.id, | ||||
|             soge_credit=True, | ||||
|             birth_date=date(2000, 1, 1), | ||||
|             gender='nonbinary', | ||||
|             clothing_cut='female', | ||||
|             clothing_size='XS', | ||||
|             health_issues='I am a bot', | ||||
|             emergency_contact_name='NoteKfet2020', | ||||
|             emergency_contact_phone='+33123456789', | ||||
|         )) | ||||
|         qs = WEIRegistration.objects.filter(user_id=user.id) | ||||
|         self.assertTrue(qs.exists()) | ||||
|         registration = qs.get() | ||||
|         self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, 200) | ||||
|         for question in WORDS: | ||||
|             # Fill 1A Survey, 20 pages | ||||
|             # be careful if questionnary form change (number of page, type of answer...) | ||||
|             response = self.client.post(reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), { | ||||
|                 question: "1" | ||||
|             }) | ||||
|             registration.refresh_from_db() | ||||
|             survey = WEISurvey2024(registration) | ||||
|             self.assertRedirects(response, reverse("wei:wei_survey", kwargs=dict(pk=registration.pk)), 302, | ||||
|                                  302 if survey.is_complete() else 200) | ||||
|             self.assertIsNotNone(getattr(survey.information, question), "Survey page " + question + " failed") | ||||
|         survey = WEISurvey2024(registration) | ||||
|         self.assertTrue(survey.is_complete()) | ||||
|         survey.select_bus(self.buses[0]) | ||||
|         survey.save() | ||||
|         self.assertIsNotNone(survey.information.get_selected_bus()) | ||||
| @@ -767,7 +767,7 @@ class TestDefaultWEISurvey(TestCase): | ||||
|         WEISurvey.update_form(None, None) | ||||
|  | ||||
|         self.assertEqual(CurrentSurvey.get_algorithm_class().get_survey_class(), CurrentSurvey) | ||||
|         self.assertEqual(CurrentSurvey.get_year(), 2024) | ||||
|         self.assertEqual(CurrentSurvey.get_year(), 2023) | ||||
|  | ||||
|  | ||||
| class TestWeiAPI(TestAPI): | ||||
|   | ||||
| @@ -114,7 +114,7 @@ msgstr "Lieu où l'activité est organisée, par exemple la Kfet." | ||||
| msgid "type" | ||||
| msgstr "type" | ||||
|  | ||||
| #: apps/activity/models.py:89 apps/logs/models.py:22 apps/member/models.py:313 | ||||
| #: apps/activity/models.py:89 apps/logs/models.py:22 apps/member/models.py:318 | ||||
| #: apps/note/models/notes.py:148 apps/treasury/models.py:293 | ||||
| #: apps/wei/models.py:171 apps/wei/templates/wei/attribute_bus_1A.html:13 | ||||
| #: apps/wei/templates/wei/survey.html:15 | ||||
| @@ -247,6 +247,7 @@ msgid "The validation of the activity is pending." | ||||
| msgstr "La validation de cette activité est en attente." | ||||
|  | ||||
| #: apps/activity/tables.py:43 apps/treasury/tables.py:107 | ||||
| #: apps/member/templates/member/picture_update.html:18 | ||||
| msgid "Remove" | ||||
| msgstr "Supprimer" | ||||
|  | ||||
| @@ -262,13 +263,13 @@ msgstr "supprimer" | ||||
| msgid "Type" | ||||
| msgstr "Type" | ||||
|  | ||||
| #: apps/activity/tables.py:84 apps/member/forms.py:193 | ||||
| #: apps/activity/tables.py:84 apps/member/forms.py:196 | ||||
| #: apps/registration/forms.py:92 apps/treasury/forms.py:131 | ||||
| #: apps/wei/forms/registration.py:104 | ||||
| msgid "Last name" | ||||
| msgstr "Nom de famille" | ||||
|  | ||||
| #: apps/activity/tables.py:86 apps/member/forms.py:198 | ||||
| #: apps/activity/tables.py:86 apps/member/forms.py:201 | ||||
| #: apps/note/templates/note/transaction_form.html:138 | ||||
| #: apps/registration/forms.py:97 apps/treasury/forms.py:133 | ||||
| #: apps/wei/forms/registration.py:109 | ||||
| @@ -400,37 +401,37 @@ msgstr "Inviter" | ||||
| msgid "Create new activity" | ||||
| msgstr "Créer une nouvelle activité" | ||||
|  | ||||
| #: apps/activity/views.py:67 note_kfet/templates/base.html:90 | ||||
| #: apps/activity/views.py:68 note_kfet/templates/base.html:90 | ||||
| msgid "Activities" | ||||
| msgstr "Activités" | ||||
|  | ||||
| #: apps/activity/views.py:93 | ||||
| #: apps/activity/views.py:108 | ||||
| msgid "Activity detail" | ||||
| msgstr "Détails de l'activité" | ||||
|  | ||||
| #: apps/activity/views.py:113 | ||||
| #: apps/activity/views.py:128 | ||||
| msgid "Update activity" | ||||
| msgstr "Modifier l'activité" | ||||
|  | ||||
| #: apps/activity/views.py:140 | ||||
| #: apps/activity/views.py:155 | ||||
| msgid "Invite guest to the activity \"{}\"" | ||||
| msgstr "Invitation pour l'activité « {} »" | ||||
|  | ||||
| #: apps/activity/views.py:178 | ||||
| #: apps/activity/views.py:193 | ||||
| msgid "You are not allowed to display the entry interface for this activity." | ||||
| msgstr "" | ||||
| "Vous n'êtes pas autorisé·e à afficher l'interface des entrées pour cette " | ||||
| "activité." | ||||
|  | ||||
| #: apps/activity/views.py:181 | ||||
| #: apps/activity/views.py:196 | ||||
| msgid "This activity does not support activity entries." | ||||
| msgstr "Cette activité ne requiert pas d'entrées." | ||||
|  | ||||
| #: apps/activity/views.py:184 | ||||
| #: apps/activity/views.py:199 | ||||
| msgid "This activity is closed." | ||||
| msgstr "Cette activité est fermée." | ||||
|  | ||||
| #: apps/activity/views.py:280 | ||||
| #: apps/activity/views.py:295 | ||||
| msgid "Entry for activity \"{}\"" | ||||
| msgstr "Entrées pour l'activité « {} »" | ||||
|  | ||||
| @@ -507,11 +508,11 @@ msgstr "cotisation pour adhérer (normalien·ne élève)" | ||||
| msgid "membership fee (unpaid students)" | ||||
| msgstr "cotisation pour adhérer (normalien·ne étudiant·e)" | ||||
|  | ||||
| #: apps/member/admin.py:65 apps/member/models.py:325 | ||||
| #: apps/member/admin.py:65 apps/member/models.py:330 | ||||
| msgid "roles" | ||||
| msgstr "rôles" | ||||
|  | ||||
| #: apps/member/admin.py:66 apps/member/models.py:339 | ||||
| #: apps/member/admin.py:66 apps/member/models.py:344 | ||||
| msgid "fee" | ||||
| msgstr "cotisation" | ||||
|  | ||||
| @@ -563,29 +564,29 @@ msgid "This image cannot be loaded." | ||||
| msgstr "Cette image ne peut pas être chargée." | ||||
|  | ||||
| #: apps/member/forms.py:148 apps/member/views.py:102 | ||||
| #: apps/registration/forms.py:34 apps/registration/views.py:266 | ||||
| #: apps/registration/forms.py:34 apps/registration/views.py:276 | ||||
| msgid "An alias with a similar name already exists." | ||||
| msgstr "Un alias avec un nom similaire existe déjà." | ||||
|  | ||||
| #: apps/member/forms.py:172 | ||||
| #: apps/member/forms.py:175 | ||||
| msgid "Inscription paid by Société Générale" | ||||
| msgstr "Inscription payée par la Société générale" | ||||
|  | ||||
| #: apps/member/forms.py:174 | ||||
| #: apps/member/forms.py:177 | ||||
| msgid "Check this case if the Société Générale paid the inscription." | ||||
| msgstr "Cochez cette case si la Société Générale a payé l'inscription." | ||||
|  | ||||
| #: apps/member/forms.py:179 apps/registration/forms.py:79 | ||||
| #: apps/member/forms.py:182 apps/registration/forms.py:79 | ||||
| #: apps/wei/forms/registration.py:91 | ||||
| msgid "Credit type" | ||||
| msgstr "Type de rechargement" | ||||
|  | ||||
| #: apps/member/forms.py:180 apps/registration/forms.py:80 | ||||
| #: apps/member/forms.py:183 apps/registration/forms.py:80 | ||||
| #: apps/wei/forms/registration.py:92 | ||||
| msgid "No credit" | ||||
| msgstr "Pas de rechargement" | ||||
|  | ||||
| #: apps/member/forms.py:182 | ||||
| #: apps/member/forms.py:185 | ||||
| msgid "You can credit the note of the user." | ||||
| msgstr "Vous pouvez créditer la note de l'utilisateur·ice avant l'adhésion." | ||||
|  | ||||
| @@ -594,17 +595,17 @@ msgstr "Vous pouvez créditer la note de l'utilisateur·ice avant l'adhésion." | ||||
| msgid "Credit amount" | ||||
| msgstr "Montant à créditer" | ||||
|  | ||||
| #: apps/member/forms.py:203 apps/note/templates/note/transaction_form.html:144 | ||||
| #: apps/member/forms.py:206 apps/note/templates/note/transaction_form.html:144 | ||||
| #: apps/registration/forms.py:102 apps/treasury/forms.py:135 | ||||
| #: apps/wei/forms/registration.py:114 | ||||
| msgid "Bank" | ||||
| msgstr "Banque" | ||||
|  | ||||
| #: apps/member/forms.py:230 | ||||
| #: apps/member/forms.py:233 | ||||
| msgid "User" | ||||
| msgstr "Utilisateur·ice" | ||||
|  | ||||
| #: apps/member/forms.py:244 | ||||
| #: apps/member/forms.py:247 | ||||
| msgid "Roles" | ||||
| msgstr "Rôles" | ||||
|  | ||||
| @@ -852,46 +853,50 @@ msgstr "" | ||||
| "Date maximale d'une fin d'adhésion, après laquelle les adhérent·e·s doivent la " | ||||
| "renouveler." | ||||
|  | ||||
| #: apps/member/models.py:263 apps/member/models.py:319 | ||||
| #: apps/member/models.py:263 | ||||
| msgid "add to registration form" | ||||
| msgstr "ajouter au formulaire d'inscription" | ||||
|  | ||||
| #: apps/member/models.py:268 apps/member/models.py:324 | ||||
| #: apps/note/models/notes.py:176 | ||||
| msgid "club" | ||||
| msgstr "club" | ||||
|  | ||||
| #: apps/member/models.py:264 | ||||
| #: apps/member/models.py:269 | ||||
| msgid "clubs" | ||||
| msgstr "clubs" | ||||
|  | ||||
| #: apps/member/models.py:330 | ||||
| #: apps/member/models.py:335 | ||||
| msgid "membership starts on" | ||||
| msgstr "l'adhésion commence le" | ||||
|  | ||||
| #: apps/member/models.py:334 | ||||
| #: apps/member/models.py:339 | ||||
| msgid "membership ends on" | ||||
| msgstr "l'adhésion finit le" | ||||
|  | ||||
| #: apps/member/models.py:343 apps/note/models/transactions.py:385 | ||||
| #: apps/member/models.py:348 apps/note/models/transactions.py:385 | ||||
| msgid "membership" | ||||
| msgstr "adhésion" | ||||
|  | ||||
| #: apps/member/models.py:344 | ||||
| #: apps/member/models.py:349 | ||||
| msgid "memberships" | ||||
| msgstr "adhésions" | ||||
|  | ||||
| #: apps/member/models.py:348 | ||||
| #: apps/member/models.py:353 | ||||
| #, python-brace-format | ||||
| msgid "Membership of {user} for the club {club}" | ||||
| msgstr "Adhésion de {user} pour le club {club}" | ||||
|  | ||||
| #: apps/member/models.py:367 | ||||
| #: apps/member/models.py:372 | ||||
| #, python-brace-format | ||||
| msgid "The role {role} does not apply to the club {club}." | ||||
| msgstr "Le rôle {role} ne s'applique pas au club {club}." | ||||
|  | ||||
| #: apps/member/models.py:376 apps/member/views.py:712 | ||||
| #: apps/member/models.py:381 apps/member/views.py:712 | ||||
| msgid "User is already a member of the club" | ||||
| msgstr "L'utilisateur·ice est déjà membre du club" | ||||
|  | ||||
| #: apps/member/models.py:388 apps/member/views.py:721 | ||||
| #: apps/member/models.py:393 apps/member/views.py:721 | ||||
| msgid "User is not a member of the parent club" | ||||
| msgstr "L'utilisateur·ice n'est pas membre du club parent" | ||||
|  | ||||
| @@ -1153,11 +1158,11 @@ msgstr "Introspection :" | ||||
| msgid "Show my applications" | ||||
| msgstr "Voir mes applications" | ||||
|  | ||||
| #: apps/member/templates/member/picture_update.html:35 | ||||
| #: apps/member/templates/member/picture_update.html:38 | ||||
| msgid "Nevermind" | ||||
| msgstr "Annuler" | ||||
|  | ||||
| #: apps/member/templates/member/picture_update.html:36 | ||||
| #: apps/member/templates/member/picture_update.html:39 | ||||
| msgid "Crop and upload" | ||||
| msgstr "Recadrer et envoyer" | ||||
|  | ||||
| @@ -1696,6 +1701,8 @@ msgstr "Historique des transactions récentes" | ||||
| #: apps/note/templates/note/mails/negative_balance.txt:25 | ||||
| #: apps/note/templates/note/mails/negative_notes_report.html:46 | ||||
| #: apps/note/templates/note/mails/negative_notes_report.txt:13 | ||||
| #: apps/note/templates/note/mails/summary_notes_report.html:62 | ||||
| #: apps/note/templates/note/mails/summary_ntoes_report.txt:33 | ||||
| #: apps/note/templates/note/mails/weekly_report.html:51 | ||||
| #: apps/note/templates/note/mails/weekly_report.txt:32 | ||||
| #: apps/registration/templates/registration/mails/email_validation_email.html:40 | ||||
| @@ -2183,18 +2190,23 @@ msgstr "Utilisateur·ice·s en attente d'inscription" | ||||
| msgid "Registration detail" | ||||
| msgstr "Détails de l'inscription" | ||||
|  | ||||
| #: apps/registration/views.py:293 | ||||
| #: apps/registration/views.py:256 | ||||
| #, python-format | ||||
| msgid "Join %(club)s Club" | ||||
| msgstr "Adhérer au club %(club)s" | ||||
|  | ||||
| #: apps/registration/views.py:299 | ||||
| msgid "You must join the BDE." | ||||
| msgstr "Vous devez adhérer au BDE." | ||||
|  | ||||
| #: apps/registration/views.py:323 | ||||
| #: apps/registration/views.py:330 | ||||
| msgid "" | ||||
| "The entered amount is not enough for the memberships, should be at least {}" | ||||
| msgstr "" | ||||
| "Le montant crédité est trop faible pour adhérer, il doit être au minimum de " | ||||
| "{}" | ||||
|  | ||||
| #: apps/registration/views.py:417 | ||||
| #: apps/registration/views.py:425 | ||||
| msgid "Invalidate pre-registration" | ||||
| msgstr "Invalider l'inscription" | ||||
|  | ||||
| @@ -2252,6 +2264,7 @@ msgid "Address" | ||||
| msgstr "Adresse" | ||||
|  | ||||
| #: apps/treasury/models.py:69 apps/treasury/models.py:202 | ||||
| #: apps/treasury/models.py:472 | ||||
| msgid "Date" | ||||
| msgstr "Date" | ||||
|  | ||||
| @@ -2370,6 +2383,102 @@ msgstr "" | ||||
| "Cet·te utilisateur·ice n'a pas assez d'argent pour payer les adhésions avec sa " | ||||
| "note. Merci de lui demander de recharger sa note avant d'invalider ce crédit." | ||||
|  | ||||
| #: apps/treasury/models.py:476 | ||||
| msgid "Total positive user" | ||||
| msgstr "Nombre d'utilisateur⋅rices en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:480 | ||||
| msgid "Balance positive user" | ||||
| msgstr "Solde des utilisateur⋅rices en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:484 | ||||
| msgid "Total positive user BDE" | ||||
| msgstr "Nombre d'adhérent⋅es au BDE en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:488 | ||||
| msgid "Balance positive user BDE" | ||||
| msgstr "Solde des adhérent⋅es au BDE en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:492 | ||||
| msgid "Total zero user" | ||||
| msgstr "Nombre d'utilisateur⋅rices à zéro" | ||||
|  | ||||
| #: apps/treasury/models.py:496 | ||||
| msgid "Total zero user BDE" | ||||
| msgstr "Nombre d'adhérent⋅es au BDE à zéro" | ||||
|  | ||||
| #: apps/treasury/models.py:500 | ||||
| msgid "Total negative user" | ||||
| msgstr "Nombre d'utilisateur⋅rices en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:504 | ||||
| msgid "Balance negative user" | ||||
| msgstr "Solde des utilisateur⋅rices en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:508 | ||||
| msgid "Total negative user BDE" | ||||
| msgstr "Nombre d'adhérent⋅es au BDE en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:512 | ||||
| msgid "Balance negative user BDE" | ||||
| msgstr "Solde des adhérent⋅es au BDE en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:516 | ||||
| msgid "Total very negative user" | ||||
| msgstr "Nombre d'utilisateur⋅rices en négatif sévère" | ||||
|  | ||||
| #: apps/treasury/models.py:520 | ||||
| msgid "Balance very negative user" | ||||
| msgstr "Solde des utilisateur⋅rices en négatif sévère" | ||||
|  | ||||
| #: apps/treasury/models.py:524 | ||||
| msgid "Total very negative user BDE" | ||||
| msgstr "Nombre d'adhérent⋅es au BDE en négatif sévère" | ||||
|  | ||||
| #: apps/treasury/models.py:528 | ||||
| msgid "Balance very negative user BDE" | ||||
| msgstr "Solde des adhérent⋅es au BDE en négatif sévère" | ||||
|  | ||||
| #: apps/treasury/models.py:532 | ||||
| msgid "Total positive club" | ||||
| msgstr "Nombre de clubs en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:536 | ||||
| msgid "Balance positive club" | ||||
| msgstr "Solde des clubs en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:540 | ||||
| msgid "Total positive club nbde" | ||||
| msgstr "Nombre de clubs non-BDE en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:544 | ||||
| msgid "Balance positive club nbde" | ||||
| msgstr "Solde des clubs non-BDE en positif" | ||||
|  | ||||
| #: apps/treasury/models.py:548 | ||||
| msgid "Total zero club" | ||||
| msgstr "Nombre de clubs à zéro" | ||||
|  | ||||
| #: apps/treasury/models.py:552 | ||||
| msgid "Total zero club nbde" | ||||
| msgstr "Nombre de clubs non-BDE à zéro" | ||||
|  | ||||
| #: apps/treasury/models.py:556 | ||||
| msgid "Total negative club" | ||||
| msgstr "Nombre de clubs en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:560 | ||||
| msgid "Balance negative club" | ||||
| msgstr "Solde des clubs en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:564 | ||||
| msgid "Total negative club nbde" | ||||
| msgstr "Nombre de clubs non-BDE en négatif" | ||||
|  | ||||
| #: apps/treasury/models.py:568 | ||||
| msgid "Balance negative club nbde" | ||||
| msgstr "Solde des clubs non-BDE en négatif" | ||||
|  | ||||
| #: apps/treasury/tables.py:20 | ||||
| msgid "Invoice #{:d}" | ||||
| msgstr "Facture n°{:d}" | ||||
| @@ -3620,9 +3729,6 @@ msgstr "" | ||||
| "d'adhésion. Vous devez également valider votre adresse email en suivant le " | ||||
| "lien que vous avez reçu." | ||||
|  | ||||
| #~ msgid "Join BDA Club" | ||||
| #~ msgstr "Adhérer au club BDA" | ||||
|  | ||||
| #, fuzzy | ||||
| #~| msgid "People having you as a friend" | ||||
| #~ msgid "You already have that person as a friend" | ||||
|   | ||||
| @@ -19,6 +19,8 @@ MAILTO=notekfet2020@lists.crans.org | ||||
|  00  5     *   *   2     root   cd /var/www/note_kfet && env/bin/python manage.py send_mail_to_negative_balances --spam --negative-amount 1 -v 0 | ||||
| # Envoyer le rapport mensuel aux trésoriers et respos info | ||||
|  00  8     *   *   5     root   cd /var/www/note_kfet && env/bin/python manage.py send_mail_to_negative_balances --report --add-years 1 -v 0 | ||||
| # Envoyer le recap de tresorerie | ||||
|  00  8     *   *   5     root   cd /var/www/note_kfet && env/bin/python manage.py send_summary_notes_report --negative-amount 2000 | ||||
| # Envoyer les rapports aux gens | ||||
|  55  6     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py send_reports -v 0 | ||||
| # Mettre à jour les boutons mis en avant | ||||
|   | ||||
		Reference in New Issue
	
	Block a user