mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-25 22:23:09 +02:00 
			
		
		
		
	Compare commits
	
		
			50 Commits
		
	
	
		
			cc2bfbe2a0
			...
			potvieux
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 9dc11aa016 | ||
|  | ff4353d344 | ||
|  | a90f45bd8b | ||
|  | 10c22ccc53 | ||
|  | 6969cee0f3 | ||
|  | ddeada200b | ||
|  | 8e2b24b2da | ||
|  | bd76c280ec | ||
|  | ca0a95ba9e | ||
|  | 614f76e699 | ||
|  | a5815f0bc7 | ||
|  | 84e9fea15f | ||
|  | b7a660ee40 | ||
|  | b9ebb1718a | ||
|  | 7ba5c76a89 | ||
|  | 702ddb5679 | ||
|  | 93aed87265 | ||
|  | 60355196ce | ||
|  | 9bffb32a5e | ||
|  | 5ef019c5c2 | ||
|  | 8da62e62fb | ||
|  | 56a43396d4 | ||
|  | 7966d6f397 | ||
|  | cb61c511ce | ||
|  | 25bfa575ed | ||
|  | e21d9fcfbe | ||
|  | b293904525 | ||
|  | bd7e6b8ad4 | ||
|  | 562dcfb908 | ||
|  | 12ef258ff0 | ||
|  | 2ae32ee3b6 | ||
|  | ec1bd45481 | ||
|  | 6c63c6417c | ||
|  | 4563b2b640 | ||
|  | 2ec5a0d9ca | ||
|  | 3faf611816 | ||
|  | 2807b6ef44 | ||
|  | d6645900d3 | ||
|  | e35847ebd8 | ||
|  | 57268bc9c2 | ||
|  | ab6c943126 | ||
|  | 5dc5f56ae4 | ||
|  | 81017fc393 | ||
|  | 90e3871934 | ||
|  | 95e07f3148 | ||
|  | 770c748bd9 | ||
|  | 06fa096405 | ||
|  | 182f680507 | ||
|  | 5ceda66ded | ||
|  | 6443d64b69 | 
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							| @@ -58,7 +58,13 @@ Bien que cela permette de créer une instance sur toutes les distributions, | ||||
|     (env)$ ./manage.py createsuperuser  # Création d'un⋅e utilisateur⋅rice initial | ||||
|     ``` | ||||
|  | ||||
| 6.  Enjoy : | ||||
| 6. (Optionnel) **Création d'une clé privée OpenID Connect** | ||||
|  | ||||
| Pour activer le support d'OpenID Connect, il faut générer une clé privée, par | ||||
| exemple avec openssl (`openssl genrsa -out oidc.key 4096`), et renseigner son | ||||
| emplacement dans `OIDC_RSA_PRIVATE_KEY` (par défaut `/var/secrets/oidc.key`). | ||||
|  | ||||
| 7.  Enjoy : | ||||
|  | ||||
|     ```bash | ||||
|     (env)$ ./manage.py runserver 0.0.0.0:8000 | ||||
| @@ -228,7 +234,13 @@ Sinon vous pouvez suivre les étapes décrites ci-dessous. | ||||
|         (env)$ ./manage.py check # pas de bêtise qui traine | ||||
|         (env)$ ./manage.py migrate | ||||
|  | ||||
| 7.  *Enjoy \o/* | ||||
| 7. **Création d'une clé privée OpenID Connect** | ||||
|  | ||||
| Pour activer le support d'OpenID Connect, il faut générer une clé privée, par | ||||
| exemple avec openssl (`openssl genrsa -out oidc.key 4096`), et renseigner son | ||||
| emplacement dans `OIDC_RSA_PRIVATE_KEY` (par défaut `/var/secrets/oidc.key`). | ||||
|  | ||||
| 8.  *Enjoy \o/* | ||||
|  | ||||
| ### Installation avec Docker | ||||
|  | ||||
|   | ||||
| @@ -35,7 +35,7 @@ class GuestAdmin(admin.ModelAdmin): | ||||
|     """ | ||||
|     Admin customisation for Guest | ||||
|     """ | ||||
|     list_display = ('last_name', 'first_name', 'activity', 'inviter') | ||||
|     list_display = ('last_name', 'first_name', 'school', 'activity', 'inviter') | ||||
|     form = GuestForm | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -51,9 +51,9 @@ class GuestViewSet(ReadProtectedModelViewSet): | ||||
|     queryset = Guest.objects.order_by('id') | ||||
|     serializer_class = GuestSerializer | ||||
|     filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter] | ||||
|     filterset_fields = ['activity', 'activity__name', 'last_name', 'first_name', 'inviter', 'inviter__alias__name', | ||||
|     filterset_fields = ['activity', 'activity__name', 'last_name', 'first_name', 'school', 'inviter', 'inviter__alias__name', | ||||
|                         'inviter__alias__normalized_name', ] | ||||
|     search_fields = ['$activity__name', '$last_name', '$first_name', '$inviter__user__email', '$inviter__alias__name', | ||||
|     search_fields = ['$activity__name', '$last_name', '$first_name', '$school', '$inviter__user__email', '$inviter__alias__name', | ||||
|                      '$inviter__alias__normalized_name', ] | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -107,7 +107,7 @@ class GuestForm(forms.ModelForm): | ||||
|  | ||||
|     class Meta: | ||||
|         model = Guest | ||||
|         fields = ('last_name', 'first_name', 'inviter', ) | ||||
|         fields = ('last_name', 'first_name', 'school', 'inviter', ) | ||||
|         widgets = { | ||||
|             "inviter": Autocomplete( | ||||
|                 NoteUser, | ||||
|   | ||||
							
								
								
									
										18
									
								
								apps/activity/migrations/0006_guest_school.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/activity/migrations/0006_guest_school.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Generated by Django 4.2.20 on 2025-03-25 09:58 | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|     dependencies = [ | ||||
|         ("activity", "0005_alter_opener_options_alter_opener_opener"), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AddField( | ||||
|             model_name="guest", | ||||
|             name="school", | ||||
|             field=models.CharField(default="", max_length=255, verbose_name="school"), | ||||
|             preserve_default=False, | ||||
|         ), | ||||
|     ] | ||||
| @@ -201,7 +201,8 @@ class Entry(models.Model): | ||||
|     def save(self, *args, **kwargs): | ||||
|         qs = Entry.objects.filter(~Q(pk=self.pk), activity=self.activity, note=self.note, guest=self.guest) | ||||
|         if qs.exists(): | ||||
|             raise ValidationError(_("Already entered on ") + _("{:%Y-%m-%d %H:%M:%S}").format(qs.get().time, )) | ||||
|             raise ValidationError(_("Already entered on ") | ||||
|                                   + _("{:%Y-%m-%d %H:%M:%S}").format(timezone.localtime(qs.get().time), )) | ||||
|  | ||||
|         if self.guest: | ||||
|             self.note = self.guest.inviter | ||||
| @@ -247,6 +248,11 @@ class Guest(models.Model): | ||||
|         verbose_name=_("first name"), | ||||
|     ) | ||||
|  | ||||
|     school = models.CharField( | ||||
|         max_length=255, | ||||
|         verbose_name=_("school"), | ||||
|     ) | ||||
|  | ||||
|     inviter = models.ForeignKey( | ||||
|         NoteUser, | ||||
|         on_delete=models.PROTECT, | ||||
|   | ||||
| @@ -1,6 +1,8 @@ | ||||
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| import datetime | ||||
|  | ||||
| from django.utils import timezone | ||||
| from django.utils.html import escape | ||||
| from django.utils.safestring import mark_safe | ||||
| @@ -51,11 +53,11 @@ class GuestTable(tables.Table): | ||||
|         } | ||||
|         model = Guest | ||||
|         template_name = 'django_tables2/bootstrap4.html' | ||||
|         fields = ("last_name", "first_name", "inviter", ) | ||||
|         fields = ("last_name", "first_name", "inviter", "school") | ||||
|  | ||||
|     def render_entry(self, record): | ||||
|         if record.has_entry: | ||||
|             return str(_("Entered on ") + str(_("{:%Y-%m-%d %H:%M:%S}").format(record.entry.time, ))) | ||||
|             return str(_("Entered on ") + str(_("{:%Y-%m-%d %H:%M:%S}").format(timezone.localtime(record.entry.time)))) | ||||
|         return mark_safe('<button id="{id}" class="btn btn-danger btn-sm" onclick="remove_guest(this.id)"> ' | ||||
|                          '{delete_trans}</button>'.format(id=record.id, delete_trans=_("remove").capitalize())) | ||||
|  | ||||
| @@ -77,6 +79,9 @@ def get_row_class(record): | ||||
|             c += " table-info" | ||||
|         elif record.note.balance < 0: | ||||
|             c += " table-danger" | ||||
|         # MODE VIEUXCON=ON | ||||
|         if (datetime.datetime.utcnow().timestamp() - record.note.created_at.timestamp()) > 3600 * 24 * 365 * 2.5: | ||||
|             c += " font-weight-bold underline" | ||||
|     return c | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -50,6 +50,7 @@ class TestActivities(TestCase): | ||||
|             inviter=self.user.note, | ||||
|             last_name="GUEST", | ||||
|             first_name="Guest", | ||||
|             school="School", | ||||
|         ) | ||||
|  | ||||
|     def test_activity_list(self): | ||||
| @@ -156,6 +157,7 @@ class TestActivities(TestCase): | ||||
|             inviter=self.user.note.id, | ||||
|             last_name="GUEST2", | ||||
|             first_name="Guest", | ||||
|             school="School", | ||||
|         )) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
| @@ -167,6 +169,7 @@ class TestActivities(TestCase): | ||||
|             inviter=self.user.note.id, | ||||
|             last_name="GUEST2", | ||||
|             first_name="Guest", | ||||
|             school="School", | ||||
|         )) | ||||
|         self.assertRedirects(response, reverse("activity:activity_detail", args=(self.activity.pk,)), 302, 200) | ||||
|  | ||||
| @@ -200,6 +203,7 @@ class TestActivityAPI(TestAPI): | ||||
|             inviter=self.user.note, | ||||
|             last_name="GUEST", | ||||
|             first_name="Guest", | ||||
|             school="School", | ||||
|         ) | ||||
|  | ||||
|         self.entry = Entry.objects.create( | ||||
|   | ||||
| @@ -168,6 +168,7 @@ class ActivityInviteView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|             activity=activity, | ||||
|             first_name="", | ||||
|             last_name="", | ||||
|             school="", | ||||
|             inviter=self.request.user.note, | ||||
|         ) | ||||
|  | ||||
| @@ -263,13 +264,22 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView): | ||||
|                                          balance=F("note__balance")) | ||||
|  | ||||
|         # Keep only users that have a note | ||||
|         note_qs = note_qs.filter(note__noteuser__isnull=False) | ||||
|         note_qs = note_qs.filter(note__noteuser__isnull=False).exclude(note__inactivity_reason='forced') | ||||
|  | ||||
|         if activity.activity_type.name != "Pot Vieux": | ||||
|             # Keep only members | ||||
|             note_qs = note_qs.filter( | ||||
|                 note__noteuser__user__memberships__club=activity.attendees_club, | ||||
|                 note__noteuser__user__memberships__date_start__lte=timezone.now(), | ||||
|                 note__noteuser__user__memberships__date_end__gte=timezone.now(), | ||||
|             ) | ||||
|  | ||||
|         # Keep only valid members | ||||
|         note_qs = note_qs.filter( | ||||
|             note__noteuser__user__memberships__club=activity.attendees_club, | ||||
|             note__noteuser__user__memberships__date_start__lte=timezone.now(), | ||||
|             note__noteuser__user__memberships__date_end__gte=timezone.now()).exclude(note__inactivity_reason='forced') | ||||
| #        note_qs = note_qs.filter( | ||||
| #            note__noteuser__user__memberships__club=activity.attendees_club, | ||||
| #            note__noteuser__user__memberships__date_start__lte=timezone.now(), | ||||
| #            note__noteuser__user__memberships__date_end__gte=timezone.now(), | ||||
| #       ) | ||||
|  | ||||
|         # Filter with permission backend | ||||
|         note_qs = note_qs.filter(PermissionBackend.filter_queryset(self.request, Alias, "view")) | ||||
|   | ||||
| @@ -23,7 +23,7 @@ from .models import Profile, Club, Membership | ||||
| class CustomAuthenticationForm(AuthenticationForm): | ||||
|     permission_mask = forms.ModelChoiceField( | ||||
|         label=_("Permission mask"), | ||||
|         queryset=PermissionMask.objects.order_by("rank"), | ||||
|         queryset=PermissionMask.objects.order_by("-rank"), | ||||
|         empty_label=None, | ||||
|     ) | ||||
|  | ||||
|   | ||||
| @@ -20,12 +20,14 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
|       </form> | ||||
|     </div> | ||||
|     <!-- MODAL TO CROP THE IMAGE --> | ||||
|     <div class="modal fade" id="modalCrop"> | ||||
|     <div class="modal fade" id="modalCrop" data-backdrop="static"> | ||||
|       <div class="modal-dialog"> | ||||
|         <div class="modal-content"> | ||||
|           <div class="modal-body"> | ||||
|             <img src="" id="modal-image" style="max-width: 100%;"> | ||||
|           </div> | ||||
|             <div class="modal-body-wrapper" style="width: 500px; height: 500px; padding: 16px;"> | ||||
|               <div class="modal-body" style="width: 100%; height: 100%; padding: 0"> | ||||
|                 <img src="" id="modal-image" style="display: block; max-width: 100%;"> | ||||
|               </div> | ||||
|             </div> | ||||
|           <div class="modal-footer"> | ||||
|             <div class="btn-group pull-left" role="group"> | ||||
|               <button type="button" class="btn btn-default" id="js-zoom-in"> | ||||
|   | ||||
| @@ -89,7 +89,7 @@ SPDX-License-Identifier: GPL-2.0-or-later | ||||
|                 </ul> | ||||
|                 <div class="card-body"> | ||||
|                     <select id="debit_type" class="form-control custom-select d-none"> | ||||
|                         {% for special_type in special_types %} | ||||
|                         {% for special_type in special_types|slice:"::-1" %} | ||||
|                             <option value="{{ special_type.id }}">{{ special_type.special_type }}</option> | ||||
|                         {% endfor %} | ||||
|                     </select> | ||||
|   | ||||
| @@ -324,7 +324,7 @@ | ||||
|             "mask": 2, | ||||
|             "field": "", | ||||
|             "permanent": false, | ||||
|             "description": "Créer une transaction de ou vers la note d'un club" | ||||
|             "description": "Créer une transaction de ou vers la note d'un club tant que la source reste au dessus de -20 €" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
| @@ -3816,7 +3816,7 @@ | ||||
|             "field": "", | ||||
|             "permanent": false, | ||||
|             "description": "Créer une transaction vers la note d'un club" | ||||
|         } | ||||
| 	} | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
| @@ -4186,6 +4186,86 @@ | ||||
|             "description": "Voir la note d'un club enfant" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
|         "pk": 266, | ||||
|         "fields": { | ||||
|             "model": [ | ||||
|                 "note", | ||||
|                 "transaction" | ||||
|             ], | ||||
|             "query": "[\"OR\", {\"source_alias\": \"Carte bancaire\"}, {\"source_alias\": \"Espèces\"}, {\"source_alias\": \"Chèque\"}, {\"source_alias\": \"Virement bancaire\"}]", | ||||
|             "type": "view", | ||||
|             "mask": 2, | ||||
|             "field": "", | ||||
|             "permanent": false, | ||||
|             "description": "Voir les transactions de rechargement" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
|         "pk": 267, | ||||
|         "fields": { | ||||
|             "model": [ | ||||
|                 "note", | ||||
|                 "transaction" | ||||
|             ], | ||||
|             "query": "[\"OR\", {\"source_alias\": \"Carte bancaire\"}, {\"source_alias\": \"Espèces\"}, {\"source_alias\": \"Chèque\"}, {\"source_alias\": \"Virement bancaire\"}]", | ||||
|             "type": "change", | ||||
|             "mask": 2, | ||||
|             "field": "valid", | ||||
|             "permanent": false, | ||||
|             "description": "Mettre à jour le statut de validation d'une transaction de rechargement" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
|         "pk": 268, | ||||
|         "fields": { | ||||
|             "model": [ | ||||
|                 "note", | ||||
|                 "transaction" | ||||
|             ], | ||||
|             "query": "[\"OR\", {\"source_alias\": \"Carte bancaire\"}, {\"source_alias\": \"Espèces\"}, {\"source_alias\": \"Chèque\"}, {\"source_alias\": \"Virement bancaire\"}]", | ||||
|             "type": "change", | ||||
|             "mask": 2, | ||||
|             "field": "invalidity_reason", | ||||
|             "permanent": false, | ||||
|             "description": "Modifier la raison d'invalidité d'une transaction de rechargement" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
|         "pk": 269, | ||||
|         "fields": { | ||||
|             "model": [ | ||||
|                 "note", | ||||
|                 "transaction" | ||||
|             ], | ||||
|             "query": "[\"OR\", {\"source_alias\": \"Carte bancaire\"}, {\"source_alias\": \"Espèces\"}, {\"source_alias\": \"Chèque\"}, {\"source_alias\": \"Virement bancaire\"}]", | ||||
|             "type": "add", | ||||
|             "mask": 2, | ||||
|             "field": "", | ||||
|             "permanent": false, | ||||
|             "description": "Créer une transaction de rechargement" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.permission", | ||||
|         "pk": 270, | ||||
|         "fields": { | ||||
|             "model": [ | ||||
|                 "note", | ||||
|                 "transaction" | ||||
|             ], | ||||
|             "query": "[\"AND\", [\"OR\", {\"source\": [\"club\", \"note\"]}, {\"destination\": [\"club\", \"note\"]}], [\"OR\", {\"source__balance__gte\": {\"F\": [\"SUB\", [\"MUL\", [\"F\", \"amount\"], [\"F\", \"quantity\"]], 5000]}}, {\"valid\": false}]]", | ||||
|             "type": "add", | ||||
|             "mask": 2, | ||||
|             "field": "", | ||||
|             "permanent": false, | ||||
|             "description": "Créer une transaction de ou vers la note d'un club tant que la source reste au dessus de -50 €" | ||||
|         } | ||||
|     }, | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 1, | ||||
| @@ -4732,8 +4812,10 @@ | ||||
|                 168, | ||||
|                 176, | ||||
|                 177, | ||||
| 		178, | ||||
|                 197, | ||||
|                 211 | ||||
|                 211, | ||||
| 		244 | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
|   | ||||
							
								
								
									
										18
									
								
								apps/treasury/migrations/0010_alter_invoice_bde.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								apps/treasury/migrations/0010_alter_invoice_bde.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| # Generated by Django 4.2.20 on 2025-04-14 20:21 | ||||
|  | ||||
| from django.db import migrations, models | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ('treasury', '0009_alter_sogecredit_transactions'), | ||||
|     ] | ||||
|  | ||||
|     operations = [ | ||||
|         migrations.AlterField( | ||||
|             model_name='invoice', | ||||
|             name='bde', | ||||
|             field=models.CharField(choices=[('Diolistos', 'Diol[list]os'), ('RavePartlist', 'RavePart[list]'), ('SecretStorlist', 'SecretStor[list]'), ('TotalistSpies', 'Tota[list]Spies'), ('Saperlistpopette', 'Saper[list]popette'), ('Finalist', 'Fina[list]'), ('Listorique', '[List]orique'), ('Satellist', 'Satel[list]'), ('Monopolist', 'Monopo[list]'), ('Kataclist', 'Katac[list]')], default='Diolistos', max_length=32, verbose_name='BDE'), | ||||
|         ), | ||||
|     ] | ||||
| @@ -27,8 +27,9 @@ class Invoice(models.Model): | ||||
|  | ||||
|     bde = models.CharField( | ||||
|         max_length=32, | ||||
|         default='RavePartlist', | ||||
|         default='Diolistos', | ||||
|         choices=( | ||||
|             ('Diolistos', 'Diol[list]os'), | ||||
|             ('RavePartlist', 'RavePart[list]'), | ||||
|             ('SecretStorlist', 'SecretStor[list]'), | ||||
|             ('TotalistSpies', 'Tota[list]Spies'), | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								apps/treasury/static/img/Diolistos.png
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								apps/treasury/static/img/Diolistos.png
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.8 MiB | 
							
								
								
									
										
											BIN
										
									
								
								apps/treasury/static/img/Diolistos_bg.jpg
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								apps/treasury/static/img/Diolistos_bg.jpg
									
									
									
									
									
										Executable file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 284 KiB | 
| @@ -38,7 +38,7 @@ class Command(BaseCommand): | ||||
|             required=False, | ||||
|             help="""User will have their(s) wrapped generated, | ||||
|             all = all users | ||||
|             adh = all users who have a valid memberships to BDE during the BDE considered | ||||
|             adh = all users who have a valid cd memberships to BDE during the BDE considered | ||||
|             supersuser = all superusers | ||||
|             custom user1,user2,... = a list of username, | ||||
|             custom_id id1,id2,... = a list of user id""", | ||||
| @@ -70,15 +70,7 @@ class Command(BaseCommand): | ||||
|             dest='create', | ||||
|         ) | ||||
|  | ||||
|     def handle(self, *args, **options): | ||||
|         # useful string for output | ||||
|         red = '\033[31;1m' | ||||
|         yellow = '\033[33;1m' | ||||
|         green = '\033[32;1m' | ||||
|         abort = red + 'ABORT' | ||||
|         warning = yellow + 'WARNING' | ||||
|         success = green + 'SUCCESS' | ||||
|  | ||||
|     def handle(self, *args, **options): # NOQA | ||||
|         # Traitement des paramètres | ||||
|         verb = options['verbosity'] | ||||
|         bde = [] | ||||
| @@ -89,11 +81,11 @@ class Command(BaseCommand): | ||||
|         if options['bde_id']: | ||||
|             if bde: | ||||
|                 if verb >= 1: | ||||
|                     print(warning) | ||||
|                     print(yellow + 'You already defined bde with their name !') | ||||
|                     self.stdout.write(self.style.WARNING( | ||||
|                         "WARNING\nYou already defined bde with their name !")) | ||||
|                 if verb >= 0: | ||||
|                     print(abort) | ||||
|                 return | ||||
|                     self.stdout.write(self.style.ERROR("ABORT")) | ||||
|                 exit(1) | ||||
|             bde_id = options['bde_id'].split(',') | ||||
|             bde = [Bde.objects.get(pk=i) for i in bde_id] | ||||
|  | ||||
| @@ -113,11 +105,11 @@ class Command(BaseCommand): | ||||
|                 user = ['custom_id', [User.objects.get(pk=u) for u in user_id]] | ||||
|             else: | ||||
|                 if verb >= 1: | ||||
|                     print(warning) | ||||
|                     print(yellow + 'You user option is not recognized') | ||||
|                     self.sdtout.write(self.style.WARNING( | ||||
|                         "WARNING\nYou user option is not recognized")) | ||||
|                 if verb >= 0: | ||||
|                     print(abort) | ||||
|                 return | ||||
|                     self.stdout.write(self.style.ERROR("ABORT")) | ||||
|                 exit(1) | ||||
|  | ||||
|         club = [] | ||||
|         if options['club']: | ||||
| @@ -133,11 +125,11 @@ class Command(BaseCommand): | ||||
|                 club = ['custom_id', [Club.objects.get(pk=c) for c in club_id]] | ||||
|             else: | ||||
|                 if verb >= 1: | ||||
|                     print(warning) | ||||
|                     print(yellow + 'You club option is not recognized') | ||||
|                     self.stdout.write(self.style.WARNING( | ||||
|                         "WARNING\nYou club option is not recognized")) | ||||
|                 if verb >= 0: | ||||
|                     print(abort) | ||||
|                 return | ||||
|                     self.stdout.write(self.style.ERROR("ABORT")) | ||||
|                 exit(1) | ||||
|  | ||||
|         change = options['change'] | ||||
|         create = options['create'] | ||||
| @@ -145,72 +137,75 @@ class Command(BaseCommand): | ||||
|         # check if parameters are sufficient for generate wrapped with the desired option | ||||
|         if not bde: | ||||
|             if verb >= 1: | ||||
|                 print(warning) | ||||
|                 print(yellow + 'You have not selectionned a BDE !') | ||||
|                 self.stdout.write(self.style.WARNING( | ||||
|                     "WARNING\nYou have not selectionned a BDE !")) | ||||
|             if verb >= 0: | ||||
|                 print(abort) | ||||
|             return | ||||
|                 self.stdout.write(self.style.ERROR("ABORT")) | ||||
|             exit(1) | ||||
|         if not (user or club): | ||||
|             if verb >= 1: | ||||
|                 print(warning) | ||||
|                 print(yellow + 'No club or user selected !') | ||||
|                 self.stdout.write(self.style.WARNING( | ||||
|                     "WARNING\nNo club or user selected !")) | ||||
|             if verb >= 0: | ||||
|                 print(abort) | ||||
|             return | ||||
|                 self.stdout.write(self.style.ERROR("ABORT")) | ||||
|             exit(1) | ||||
|  | ||||
|         if verb >= 3: | ||||
|             print('\033[1mOptions:\033[m') | ||||
|             self.stdout.write("Options:") | ||||
|             bde_str = '' | ||||
|             for b in bde: | ||||
|                 bde_str += str(b) | ||||
|             print('BDE: ' + bde_str) | ||||
|                 bde_str += str(b) + '\n' | ||||
|             self.stdout.write("BDE: " + bde_str) | ||||
|             if user: | ||||
|                 print('User: ' + user[0]) | ||||
|                 self.stdout.write('User: ' + user[0]) | ||||
|             if club: | ||||
|                 print('Club: ' + club[0]) | ||||
|             print('change: ' + str(change)) | ||||
|             print('create: ' + str(create)) | ||||
|             print('') | ||||
|                 self.stdout.write('Club: ' + club[0]) | ||||
|             self.stdout.write('change: ' + str(change)) | ||||
|             self.stdout.write('create: ' + str(create) + '\n') | ||||
|         if not (change or create): | ||||
|             if verb >= 1: | ||||
|                 print(warning) | ||||
|                 print(yellow + 'change and create is set to false, none wrapped will be created') | ||||
|                 self.stdout.write(self.style.WARNING( | ||||
|                     "WARNING\nchange and create is set to false, none wrapped will be created")) | ||||
|             if verb >= 0: | ||||
|                 print(abort) | ||||
|             return | ||||
|                 self.stdout.write(self.style.ERROR("ABORT")) | ||||
|             exit(1) | ||||
|         if verb >= 1 and change: | ||||
|             print(warning) | ||||
|             print(yellow + 'change is set to true, some wrapped may be replaced !') | ||||
|             self.stdout.write(self.style.WARNING( | ||||
|                 "WARNING\nchange is set to true, some wrapped may be replaced !")) | ||||
|         if verb >= 1 and not create: | ||||
|             print(warning) | ||||
|             print(yellow + 'create is set to false, wrapped will not be created !') | ||||
|             self.stdout.write(self.style.WARNING( | ||||
|                 "WARNING\ncreate is set to false, wrapped will not be created !")) | ||||
|         if verb >= 3 or change or not create: | ||||
|             a = str(input('\033[mContinue ? (y/n) ')).lower() | ||||
|             if a in ['n', 'no', 'non', '0']: | ||||
|                 if verb >= 0: | ||||
|                     print(abort) | ||||
|                 return | ||||
|                     self.stdout.write(self.style.ERROR("ABORT")) | ||||
|                 exit(1) | ||||
|  | ||||
|         note = self.convert_to_note(change, create, bde=bde, user=user, club=club, verb=verb) | ||||
|         if verb >= 1: | ||||
|             print("\033[32mUser and/or Club given has successfully convert in their note\033[m") | ||||
|             self.stdout.write(self.style.SUCCESS( | ||||
|                 "User and/or Club given has successfully convert in their note")) | ||||
|  | ||||
|         global_data = self.global_data(bde, verb=verb) | ||||
|         if verb >= 1: | ||||
|             print("\033[32mGlobal data has been successfully generated\033[m") | ||||
|             self.stdout.write(self.style.SUCCESS( | ||||
|                 "Global data has been successfully generated")) | ||||
|  | ||||
|         unique_data = self.unique_data(bde, note, global_data=global_data, verb=verb) | ||||
|         if verb >= 1: | ||||
|             print("\033[32mUnique data has been successfully generated\033[m") | ||||
|             self.stdout.write(self.style.SUCCESS( | ||||
|                 "Unique data has been successfully generated")) | ||||
|  | ||||
|         self.make_wrapped(unique_data, note, bde, change, create, verb=verb) | ||||
|         if verb >= 1: | ||||
|             print(green + "The wrapped has been generated !") | ||||
|             self.stdout.write(self.style.SUCCESS( | ||||
|                 "The wrapped has been generated !")) | ||||
|         if verb >= 0: | ||||
|             print(success) | ||||
|             self.stdout.write(self.style.SUCCESS("SUCCESS")) | ||||
|         exit(0) | ||||
|  | ||||
|         return | ||||
|  | ||||
|     def convert_to_note(self, change, create, bde=None, user=None, club=None, verb=1): | ||||
|     def convert_to_note(self, change, create, bde=None, user=None, club=None, verb=1): # NOQA | ||||
|         notes = [] | ||||
|         for b in bde: | ||||
|             note_for_bde = Note.objects.filter(pk__lte=-1) | ||||
| @@ -253,17 +248,17 @@ class Command(BaseCommand): | ||||
|             note_for_bde = self.filter_note(b, note_for_bde, change, create, verb=verb) | ||||
|             notes.append(note_for_bde) | ||||
|             if verb >= 2: | ||||
|                 print("\033[m{nb} note selectionned for bde {bde}".format(nb=len(note_for_bde), bde=b.name)) | ||||
|                 self.stdout.write(f"{len(note_for_bde)} note selectionned for bde {b.name}") | ||||
|         return notes | ||||
|  | ||||
|     def global_data(self, bde, verb=1): | ||||
|     def global_data(self, bde, verb=1): # NOQA | ||||
|         data = {} | ||||
|         for b in bde: | ||||
|             if b.name == 'Rave Part[list]': | ||||
|                 if verb >= 2: | ||||
|                     print("Begin to make global data") | ||||
|                     self.stdout.write("Begin to make global data") | ||||
|                 if verb >= 3: | ||||
|                     print('nb_transaction') | ||||
|                     self.stdout.write("nb_transaction") | ||||
|                 # nb total de transactions | ||||
|                 data['nb_transaction'] = Transaction.objects.filter( | ||||
|                     created_at__gte=b.date_start, | ||||
| @@ -271,7 +266,7 @@ class Command(BaseCommand): | ||||
|                     valid=True).count() | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('nb_vieux_con') | ||||
|                     self.stdout.write("nb_vieux_con") | ||||
|                 # nb total de vielleux con·ne·s derrière le bar | ||||
|                 button_id = [2884, 2585] | ||||
|                 transactions = Transaction.objects.filter( | ||||
| @@ -286,7 +281,7 @@ class Command(BaseCommand): | ||||
|                 data['nb_vieux_con'] = q | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('nb_soiree') | ||||
|                     self.stdout.write("nb_soiree") | ||||
|                 # nb total de soirée | ||||
|                 a_type_id = [1, 2, 4, 5, 7, 10] | ||||
|                 data['nb_soiree'] = Activity.objects.filter( | ||||
| @@ -296,7 +291,7 @@ class Command(BaseCommand): | ||||
|                     activity_type__pk__in=a_type_id).count() | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('pots, nb_entree_pot') | ||||
|                     self.stdout.write('pots, nb_entree_pot') | ||||
|                 # nb d'entrée totale aux pots | ||||
|                 pot_id = [1, 4, 10] | ||||
|                 pots = Activity.objects.filter( | ||||
| @@ -310,7 +305,7 @@ class Command(BaseCommand): | ||||
|                     data['nb_entree_pot'] += Entry.objects.filter(activity=pot).count() | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('top3_buttons') | ||||
|                     self.stdout.write('top3_buttons') | ||||
|                 # top 3 des boutons les plus cliqués | ||||
|                 transactions = Transaction.objects.filter( | ||||
|                     created_at__gte=b.date_start, | ||||
| @@ -329,7 +324,7 @@ class Command(BaseCommand): | ||||
|                 data['top3_buttons'] = list(sorted(d.items(), key=lambda item: item[1], reverse=True))[:3] | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('class_conso_all') | ||||
|                     self.stdout.write('class_conso_all') | ||||
|                 # le classement des plus gros consommateurs (BDE + club) | ||||
|                 transactions = Transaction.objects.filter( | ||||
|                     created_at__gte=b.date_start, | ||||
| @@ -348,7 +343,7 @@ class Command(BaseCommand): | ||||
|                 data['class_conso_all'] = dict(sorted(d.items(), key=lambda item: item[1], reverse=True)) | ||||
|  | ||||
|                 if verb >= 3: | ||||
|                     print('class_conso_bde') | ||||
|                     self.stdout.write('class_conso_bde') | ||||
|                 # le classement des plus gros consommateurs BDE | ||||
|                 transactions = Transaction.objects.filter( | ||||
|                     created_at__gte=b.date_start, | ||||
| @@ -368,11 +363,10 @@ class Command(BaseCommand): | ||||
|  | ||||
|             else: | ||||
|                 # make your wrapped or reuse previous wrapped | ||||
|                 raise NotImplementedError("The BDE: {bde_name} has not personalized wrapped, make it !" | ||||
|                                           .format(bde_name=b.name)) | ||||
|                 raise NotImplementedError(f"The BDE: {b.name} has not personalized wrapped, make it !") | ||||
|         return data | ||||
|  | ||||
|     def unique_data(self, bde, note, global_data=None, verb=1): | ||||
|     def unique_data(self, bde, note, global_data=None, verb=1): # NOQA | ||||
|         data = [] | ||||
|         for i in range(len(bde)): | ||||
|             data_bde = [] | ||||
| @@ -380,8 +374,7 @@ class Command(BaseCommand): | ||||
|                 if verb >= 3: | ||||
|                     total = len(note[i]) | ||||
|                     current = 0 | ||||
|                     print('Make {nb} data for wrapped sponsored by {bde}' | ||||
|                           .format(nb=total, bde=bde[i].name)) | ||||
|                     self.stdout.write(f"Make {total} data for wrapped sponsored by {bde[i].name}") | ||||
|                 for n in note[i]: | ||||
|                     d = {} | ||||
|                     if 'user' in n.__dir__(): | ||||
| @@ -542,12 +535,11 @@ class Command(BaseCommand): | ||||
|                     data_bde.append(json.dumps(d)) | ||||
|                     if verb >= 3: | ||||
|                         current += 1 | ||||
|                         print('\033[2K' + '({c}/{t})'.format(c=current, t=total) + '\033[1A') | ||||
|                         self.stdout.write("\033[2K" + f"({current}/{total})" + "\033[1A") | ||||
|  | ||||
|             else: | ||||
|                 # make your wrapped or reuse previous wrapped | ||||
|                 raise NotImplementedError("The BDE: {bde_name} has not personalized wrapped, make it !" | ||||
|                                           .format(bde_name=bde[i].name)) | ||||
|                 raise NotImplementedError(f"The BDE: {bde[i].name} has not personalized wrapped, make it !") | ||||
|             data.append(data_bde) | ||||
|         return data | ||||
|  | ||||
| @@ -557,7 +549,7 @@ class Command(BaseCommand): | ||||
|             total = 0 | ||||
|             for n in note: | ||||
|                 total += len(n) | ||||
|             print('\033[mMake {nb} wrapped'.format(nb=total)) | ||||
|             self.stdout.write(f"Make {total} wrapped") | ||||
|         for i in range(len(bde)): | ||||
|             for j in range(len(note[i])): | ||||
|                 if create and not Wrapped.objects.filter(bde=bde[i], note=note[i][j]): | ||||
| @@ -572,7 +564,7 @@ class Command(BaseCommand): | ||||
|                     w.save() | ||||
|                 if verb >= 3: | ||||
|                     current += 1 | ||||
|                     print('\033[2K' + '({c}/{t})'.format(c=current, t=total) + '\033[1A') | ||||
|                     self.stdout.write("\033[2K" + f"({current}/{total})" + "\033[1A") | ||||
|         return | ||||
|  | ||||
|     def filter_note(self, bde, note, change, create, verb=1): | ||||
|   | ||||
| @@ -23,9 +23,9 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
| 		let d1 = document.getElementById("consumer"); | ||||
| 		let d2 = document.getElementById("creditor"); | ||||
| 		if (con) { d1.textContent = {{ big_consumer | safe }}[0] + " " + gettext("with") + " " + {{ big_consumer | safe}}[1] + "€";} | ||||
| 		else { d1.textContent = gettext("Infortunately, you doesn't have consumer this year");}; | ||||
| 		else { d1.textContent = gettext("{% trans "Infortunately, you doesn't have consumer this year" %}");}; | ||||
| 		if (cre) { d2.textContent = {{ big_creancier | safe}}[0] + " " + gettext("with") + " " + {{ big_creancier | safe}}[1] + "€";} | ||||
| 		else { d2.textContent = gettext("Congratulations you are a real rat !"); }; | ||||
| 		else { d2.textContent = gettext("{% trans "Congratulations you are a real rat !" %}"); }; | ||||
|  | ||||
| 	</script> | ||||
| {% endblock %} | ||||
|   | ||||
| @@ -6,17 +6,24 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="row justify-content-center">    | ||||
|     <div class="col-md-10"> | ||||
|         <div class="card card-border shadow"> | ||||
|             <div class="card-header text-center"> | ||||
| 		    <h5> {{ title }}</h5> | ||||
|             </div> | ||||
|             <div class="card-body px-0 py-0" id="wrapped_table"> | ||||
|                 {% render_table table %} | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| <div id="wrapped_tables"> | ||||
| {% if tables|length > 0 %} | ||||
| <div class="card bg-light mb-3"> | ||||
|     <h3 class="card-header text-center"> | ||||
|         {% trans "My wrapped" %} | ||||
|     </h3> | ||||
|     {% render_table tables.1 %} | ||||
| </div> | ||||
| {% endif %} | ||||
|  | ||||
| {% if tables|length > 0 %} | ||||
| <div class="card bg-light mb-3"> | ||||
|     <h3 class="card-header text-center"> | ||||
|         {% trans "Public wrapped" %} | ||||
|     </h3> | ||||
|     {% render_table tables.0 %} | ||||
| </div> | ||||
| {% endif %} | ||||
| </div> | ||||
| {% endblock %} | ||||
|  | ||||
| @@ -25,7 +32,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
| 	let club_not_public = {{ club_not_public }}; | ||||
| 	if (club_not_public) { (addMsg("{% trans "Do not forget to ask permission to people who are in your wrapped before to make them public" %}", 'warning'));} | ||||
|    function refreshTable() { | ||||
| 	$("#wrapped_table").load(location.pathname + " #wrapped_table"); | ||||
| 	$("#wrapped_tables").load(location.pathname + " #wrapped_tables"); | ||||
|    } | ||||
|  | ||||
|    function copylink(id) { | ||||
|   | ||||
							
								
								
									
										0
									
								
								apps/wrapped/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								apps/wrapped/tests/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										91
									
								
								apps/wrapped/tests/test_wrapped.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								apps/wrapped/tests/test_wrapped.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from datetime import timedelta | ||||
|  | ||||
| from api.tests import TestAPI | ||||
| from django.contrib.auth.models import User | ||||
| from django.test import TestCase | ||||
| from django.urls import reverse | ||||
| from django.utils import timezone | ||||
|  | ||||
| from ..api.views import WrappedViewSet, BdeViewSet | ||||
| from ..models import Bde, Wrapped | ||||
|  | ||||
|  | ||||
| class TestWrapped(TestCase): | ||||
|     """ | ||||
|     Test activities | ||||
|     """ | ||||
|     fixtures = ('initial',) | ||||
|  | ||||
|     def setUp(self): | ||||
|         self.user = User.objects.create_superuser( | ||||
|             username="admintoto", | ||||
|             password="tototototo", | ||||
|             email="toto@example.com" | ||||
|         ) | ||||
|         self.client.force_login(self.user) | ||||
|  | ||||
|         sess = self.client.session | ||||
|         sess["permission_mask"] = 42 | ||||
|         sess.save() | ||||
|  | ||||
|         self.bde = Bde.objects.create( | ||||
|             name="The best BDE", | ||||
|             date_start=timezone.now() - timedelta(days=365), | ||||
|             date_end=timezone.now(), | ||||
|         ) | ||||
|  | ||||
|         self.wrapped = Wrapped.objects.create( | ||||
|             generated=True, | ||||
|             public=False, | ||||
|             bde=self.bde, | ||||
|             note=self.user.note, | ||||
|             data_json="{}", | ||||
|         ) | ||||
|  | ||||
|     def test_wrapped_list(self): | ||||
|         """ | ||||
|         Display the list of all wrapped | ||||
|         """ | ||||
|         response = self.client.get(reverse("wrapped:wrapped_list")) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|     def test_wrapped_detail(self): | ||||
|         """ | ||||
|         Display the detail of an wrapped | ||||
|         """ | ||||
|         response = self.client.get(reverse("wrapped:wrapped_detail", args=(self.wrapped.pk,))) | ||||
|         self.assertEqual(response.status_code, 200) | ||||
|  | ||||
|  | ||||
| class TestWrappedAPI(TestAPI): | ||||
|     def setUp(self) -> None: | ||||
|         super().setUp() | ||||
|  | ||||
|         self.bde = Bde.objects.create( | ||||
|             name="The best BDE", | ||||
|             date_start=timezone.now() - timedelta(days=365), | ||||
|             date_end=timezone.now(), | ||||
|         ) | ||||
|  | ||||
|         self.wrapped = Wrapped.objects.create( | ||||
|             generated=True, | ||||
|             public=False, | ||||
|             bde=self.bde, | ||||
|             note=self.user.note, | ||||
|             data_json="{}", | ||||
|         ) | ||||
|  | ||||
|     def test_bde_api(self): | ||||
|         """ | ||||
|         Load Bde API page and test all filters and permissions | ||||
|         """ | ||||
|         self.check_viewset(BdeViewSet, "/api/wrapped/bde/") | ||||
|  | ||||
|     def test_wrapped_api(self): | ||||
|         """ | ||||
|         Load Wrapped API page and test all filters and permissions | ||||
|         """ | ||||
|         self.check_viewset(WrappedViewSet, "/api/wrapped/wrapped/") | ||||
| @@ -6,7 +6,8 @@ import json | ||||
| from django.contrib.auth.mixins import LoginRequiredMixin | ||||
| from django.utils.translation import gettext_lazy as _ | ||||
| from django.views.generic import DetailView | ||||
| from django_tables2.views import SingleTableView | ||||
| from django.views.generic.list import ListView | ||||
| from django_tables2.views import MultiTableMixin | ||||
| from permission.backends import PermissionBackend | ||||
| from permission.views import ProtectQuerysetMixin | ||||
|  | ||||
| @@ -14,21 +15,29 @@ from .models import Wrapped | ||||
| from .tables import WrappedTable | ||||
|  | ||||
|  | ||||
| class WrappedListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView): | ||||
| class WrappedListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView): | ||||
|     """ | ||||
|     Display all Wrapped, and classify by year | ||||
|     """ | ||||
|     model = Wrapped | ||||
|     table_class = WrappedTable | ||||
|     tables = [ | ||||
|         lambda data: WrappedTable(data, prefix="public-"), | ||||
|         lambda data: WrappedTable(data, prefix="personnal-"), | ||||
|     ] | ||||
|     template_name = 'wrapped/wrapped_list.html' | ||||
|     extra_context = {'title': _("List of wrapped")} | ||||
|  | ||||
|     def get_queryset(self, **kwargs): | ||||
|         return super().get_queryset(**kwargs).distinct() | ||||
|  | ||||
|     def get_table_data(self): | ||||
|         return Wrapped.objects.filter(PermissionBackend.filter_queryset( | ||||
|             self.request, Wrapped, "change", field='public')).distinct().order_by("-bde__date_start") | ||||
|     def get_tables_data(self): | ||||
|         return [ | ||||
|             Wrapped.objects.filter(public=True), | ||||
|             Wrapped.objects | ||||
|             .filter(PermissionBackend.filter_queryset(self.request, Wrapped, "change", field='public')) | ||||
|             .distinct() | ||||
|             .order_by("-bde__date_start") | ||||
|         ] | ||||
|  | ||||
|     def get_context_data(self, **kwargs): | ||||
|         context = super().get_context_data(**kwargs) | ||||
|   | ||||
							
								
								
									
										118
									
								
								docs/_static/img/graphs/wrapped.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								docs/_static/img/graphs/wrapped.svg
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" | ||||
|  "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd"> | ||||
| <!-- Generated by graphviz version 2.43.0 (0) | ||||
|  --> | ||||
| <!-- Title: model_graph Pages: 1 --> | ||||
| <svg width="319pt" height="245pt" | ||||
|  viewBox="0.00 0.00 319.00 245.00" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"> | ||||
| <g id="graph0" class="graph" transform="scale(1 1) rotate(0) translate(4 241)"> | ||||
| <title>model_graph</title> | ||||
| <polygon fill="white" stroke="transparent" points="-4,4 -4,-241 315,-241 315,4 -4,4"/> | ||||
| <!-- wrapped_models_Bde --> | ||||
| <g id="node1" class="node"> | ||||
| <title>wrapped_models_Bde</title> | ||||
| <polygon fill="white" stroke="transparent" points="8,-4 8,-79 158,-79 158,-4 8,-4"/> | ||||
| <polygon fill="#1b563f" stroke="transparent" points="9,-56.5 9,-77.5 157,-77.5 157,-56.5 9,-56.5"/> | ||||
| <text text-anchor="start" x="52" y="-65.5" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="62" y="-65.5" font-family="Roboto" font-weight="bold" font-size="10.00" fill="white">    Bde    </text> | ||||
| <text text-anchor="start" x="11" y="-49.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="21" y="-49.1" font-family="Roboto" font-weight="bold" font-size="8.00">id</text> | ||||
| <text text-anchor="start" x="31" y="-49.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="77" y="-49.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="87" y="-49.1" font-family="Roboto" font-weight="bold" font-size="8.00">AutoField</text> | ||||
| <text text-anchor="start" x="131" y="-49.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="11" y="-36.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="21" y="-36.1" font-family="Roboto" font-size="8.00">date_end</text> | ||||
| <text text-anchor="start" x="60" y="-36.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="77" y="-36.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="87" y="-36.1" font-family="Roboto" font-size="8.00">DateTimeField</text> | ||||
| <text text-anchor="start" x="145" y="-36.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="11" y="-23.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="21" y="-23.1" font-family="Roboto" font-size="8.00">date_start</text> | ||||
| <text text-anchor="start" x="63" y="-23.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="77" y="-23.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="87" y="-23.1" font-family="Roboto" font-size="8.00">DateTimeField</text> | ||||
| <text text-anchor="start" x="145" y="-23.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="11" y="-10.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="21" y="-10.1" font-family="Roboto" font-size="8.00">name</text> | ||||
| <text text-anchor="start" x="45" y="-10.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="77" y="-10.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="87" y="-10.1" font-family="Roboto" font-size="8.00">CharField</text> | ||||
| <text text-anchor="start" x="125" y="-10.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <polygon fill="none" stroke="black" points="8,-4 8,-79 158,-79 158,-4 8,-4"/> | ||||
| </g> | ||||
| <!-- wrapped_models_Wrapped --> | ||||
| <g id="node2" class="node"> | ||||
| <title>wrapped_models_Wrapped</title> | ||||
| <polygon fill="white" stroke="transparent" points="67,-132 67,-233 231,-233 231,-132 67,-132"/> | ||||
| <polygon fill="#1b563f" stroke="transparent" points="68,-210.5 68,-231.5 230,-231.5 230,-210.5 68,-210.5"/> | ||||
| <text text-anchor="start" x="103" y="-219.5" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="113" y="-219.5" font-family="Roboto" font-weight="bold" font-size="10.00" fill="white">    Wrapped    </text> | ||||
| <text text-anchor="start" x="70" y="-203.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-203.1" font-family="Roboto" font-weight="bold" font-size="8.00">id</text> | ||||
| <text text-anchor="start" x="90" y="-203.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-203.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-203.1" font-family="Roboto" font-weight="bold" font-size="8.00">AutoField</text> | ||||
| <text text-anchor="start" x="191" y="-203.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="70" y="-190.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-190.1" font-family="Roboto" font-weight="bold" font-size="8.00">bde</text> | ||||
| <text text-anchor="start" x="98" y="-190.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-190.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-190.1" font-family="Roboto" font-weight="bold" font-size="8.00">ForeignKey (id)</text> | ||||
| <text text-anchor="start" x="218" y="-190.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="70" y="-177.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-177.1" font-family="Roboto" font-weight="bold" font-size="8.00">note</text> | ||||
| <text text-anchor="start" x="101" y="-177.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-177.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-177.1" font-family="Roboto" font-weight="bold" font-size="8.00">ForeignKey (id)</text> | ||||
| <text text-anchor="start" x="218" y="-177.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="70" y="-164.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-164.1" font-family="Roboto" font-size="8.00">data_json</text> | ||||
| <text text-anchor="start" x="120" y="-164.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-164.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-164.1" font-family="Roboto" font-size="8.00">TextField</text> | ||||
| <text text-anchor="start" x="182" y="-164.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="70" y="-151.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-151.1" font-family="Roboto" font-size="8.00">generated</text> | ||||
| <text text-anchor="start" x="123" y="-151.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-151.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-151.1" font-family="Roboto" font-size="8.00">BooleanField</text> | ||||
| <text text-anchor="start" x="200" y="-151.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="70" y="-138.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="80" y="-138.1" font-family="Roboto" font-size="8.00">public</text> | ||||
| <text text-anchor="start" x="105" y="-138.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="137" y="-138.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <text text-anchor="start" x="147" y="-138.1" font-family="Roboto" font-size="8.00">BooleanField</text> | ||||
| <text text-anchor="start" x="200" y="-138.1" font-family="Roboto" font-size="8.00">    </text> | ||||
| <polygon fill="none" stroke="black" points="67,-132 67,-233 231,-233 231,-132 67,-132"/> | ||||
| </g> | ||||
| <!-- wrapped_models_Wrapped->wrapped_models_Bde --> | ||||
| <g id="edge1" class="edge"> | ||||
| <title>wrapped_models_Wrapped->wrapped_models_Bde</title> | ||||
| <path fill="none" stroke="black" d="M119.99,-120.4C114,-107.79 107.84,-94.82 102.31,-83.16"/> | ||||
| <ellipse fill="black" stroke="black" cx="121.77" cy="-124.15" rx="4" ry="4"/> | ||||
| <text text-anchor="middle" x="132" y="-103.6" font-family="Roboto" font-size="8.00"> bde (+)</text> | ||||
| </g> | ||||
| <!-- note_models_notes_Note --> | ||||
| <g id="node3" class="node"> | ||||
| <title>note_models_notes_Note</title> | ||||
| <polygon fill="white" stroke="transparent" points="192,-31 192,-52 240,-52 240,-31 192,-31"/> | ||||
| <polygon fill="#1b563f" stroke="transparent" points="192,-30.5 192,-51.5 240,-51.5 240,-30.5 192,-30.5"/> | ||||
| <text text-anchor="start" x="196.5" y="-38.9" font-family="Roboto" font-size="8.00">  </text> | ||||
| <text text-anchor="start" x="201.5" y="-38.9" font-family="Roboto" font-size="12.00" fill="white">Note</text> | ||||
| <text text-anchor="start" x="230.5" y="-38.9" font-family="Roboto" font-size="8.00">  </text> | ||||
| </g> | ||||
| <!-- wrapped_models_Wrapped->note_models_notes_Note --> | ||||
| <g id="edge2" class="edge"> | ||||
| <title>wrapped_models_Wrapped->note_models_notes_Note</title> | ||||
| <path fill="none" stroke="black" d="M178.48,-120.33C189.12,-98.27 200.3,-75.07 207.66,-59.8"/> | ||||
| <ellipse fill="black" stroke="black" cx="176.64" cy="-124.16" rx="4" ry="4"/> | ||||
| <text text-anchor="middle" x="204.5" y="-103.6" font-family="Roboto" font-size="8.00"> note (+)</text> | ||||
| </g> | ||||
| <!-- \n\n\n --> | ||||
| <g id="node4" class="node"> | ||||
| <title>\n\n\n</title> | ||||
| </g> | ||||
| </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 9.7 KiB | 
| @@ -55,6 +55,7 @@ Les adhérent⋅es ont la possibilité d'inviter des ami⋅es. Pour cela, les di | ||||
| * Activité concernée (clé étrangère) | ||||
| * Nom de famille | ||||
| * Prénom | ||||
| * École | ||||
| * Note de la personne ayant invité | ||||
|  | ||||
| Certaines contraintes s'appliquent : | ||||
|   | ||||
| @@ -14,6 +14,7 @@ Applications de la Note Kfet 2020 | ||||
|    logs | ||||
|    treasury | ||||
|    wei | ||||
|    wrapped | ||||
|  | ||||
| La Note Kfet 2020 est un projet Django, décomposé en applications. | ||||
| Certaines applications sont développées uniquement pour ce projet, et sont indispensables, | ||||
| @@ -69,4 +70,6 @@ Applications facultatives | ||||
|     Interface de gestion pour les trésorièr⋅es, émission de factures, remises de chèque, statistiques... | ||||
| * `WEI <wei>`_ : | ||||
|     Interface de gestion du WEI. | ||||
| * `Wrapped <wrapped>`_ : | ||||
|     Récapitulatif personnalisé annuel de statitiques globales et personnelles. | ||||
|  | ||||
|   | ||||
							
								
								
									
										108
									
								
								docs/apps/wrapped.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								docs/apps/wrapped.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | ||||
| Wrapped | ||||
| ======= | ||||
|  | ||||
| Cette application montre les statistiques annuelles des utilisateur·ice·s et/ou des clubs. | ||||
|  | ||||
| Modèles | ||||
| ------- | ||||
|  | ||||
| Bde | ||||
| ~~~ | ||||
|  | ||||
| Le modèle ``Bde`` contient des informations relatifs à un BDE : | ||||
|  | ||||
| * ``name`` : ``CharField``, nom du BDE. | ||||
| * ``date_start`` : ``DateField``, date de prise de fonction du bureau BDE considéré. | ||||
| * ``date_end`` : ``DateField``, date de démission du bureau BDE considéré. | ||||
|  | ||||
| Wrapped | ||||
| ~~~~~~~ | ||||
|  | ||||
| Contient les informations sur un wrapped : | ||||
|  | ||||
| * ``generated`` : ``BooleanField``, indique si le wrapped a été généré ou non. | ||||
| * ``public`` : ``BooleanField``, indique si le wrapped est visible de tous les utilisateur·ice·s ou non. | ||||
| * ``bde`` : ``ForeignKey(Bde)``, BDE auquel le wrapped correspond. | ||||
| * ``note`` : ``ForeignKey(Note)``, note à laquelle le wrapped correspond. | ||||
| * ``data_json`` : ``TextField``, diverses statistique concernant les notes durant le mandat BDE | ||||
|   considéré ou sur la NoteKfet dans sa globalité. | ||||
|  | ||||
| Graphe des modèles | ||||
| ~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| .. image:: ../_static/img/graphs/wrapped.svg | ||||
|    :width: 960 | ||||
|    :alt: Graphe des modèles de l'application Wrapped | ||||
|  | ||||
| Fonctionnement | ||||
| -------------- | ||||
|  | ||||
| Création d'un BDE | ||||
| ~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| Seul un⋅e respo info peut créer un BDE. Pour cela, se rendre dans l'onglet « Admin »., puis « BDE » et | ||||
| enfin « + Ajouter BDE ». Iel doit renseigner, les dates de début et de fin du bureau BDE ainsi que le | ||||
| nom de la liste. | ||||
|  | ||||
| Génération des wrappeds | ||||
| ~~~~~~~~~~~~~~~~~~~~~~~ | ||||
|  | ||||
| Seul un·e respo info peut générer des wrappeds. Pour une utilisation annuelle classique, iel exécute la | ||||
| commande : | ||||
|  | ||||
| ``./manage.py generate_wrapped -b "bde_name" -u adh -c active`` | ||||
|  | ||||
| Pour une utilisation plus technique de cette commande se référer à sa documentation | ||||
|  | ||||
| ``./manage.py help generate_wrapped`` | ||||
|  | ||||
| Le script prend une dizaine de minutes pour générer tous les wrappeds. | ||||
|  | ||||
| Créer ses propres wrappeds | ||||
| -------------------------- | ||||
|  | ||||
| Cette section est plus technique et s'addresse plutôt à des respos infos en cours de mandat qui voudrai | ||||
| faire les wrappeds de leur propre BDE. | ||||
|  | ||||
| Contenu | ||||
| ~~~~~~~ | ||||
|  | ||||
| Il est fortement conseillé de bien réfléchir à ce que l'on souhaite mettre sur un wrapped, plusieurs | ||||
| critères sont à prendre compte : | ||||
|  | ||||
| * compréhension, est-ce que la donnée fait sens auprès des utilisateur·ice·s. | ||||
| * pertinence, est-ce que la donnée fonctionne pour un grand nombre d'utilisateur. | ||||
| * faisabilité, est-ce que le temps de calcul est suffisament rapide. | ||||
| * complexité, est-ce que c'est trop compliqué à coder. | ||||
|  | ||||
| Script | ||||
| ~~~~~~ | ||||
|  | ||||
| Le script *generate_wrapped* fonctionne de la manière suivante : | ||||
|  | ||||
| * ``convert_to_note`` : en fonction des arguments d'entrée, il récupére toutes les notes dont le·s | ||||
|   wrapped·s va/vont être généré·s | ||||
|   ou regénéré·s. | ||||
| * ``global_data`` : le script génére ensuite des statistiques globales qui concernent pas qu'une seule | ||||
|  note (nombre de soirée, classement, etc).  | ||||
| * ``unique_data`` : le script génére les statitiques uniques à chaque note, et rajoute des données | ||||
|   globales si nécessaire, pour chaque note on souhaite avoir un json avec toutes les données qui | ||||
|   seront dans le wrapped. | ||||
| * ``make_wrapped`` : enfin, le cas échéant, pour chaque bde, et pour chaque note, le wrapped est crée | ||||
|   ou modifié, et enregistré, s'il est crée il est par défault non public. | ||||
|  | ||||
| Seules les fonctions ``global_data`` et ``unique_data`` sont à modifier, pour implementer un nouveau | ||||
| BDE. | ||||
|  | ||||
| Template | ||||
| ~~~~~~~~ | ||||
|  | ||||
| Il y a au moins deux templates a écrire pour chaque bde : | ||||
|  | ||||
| * ``templates/wrapped/{bde_id}/wrapped_view_club.html``: le template pour les wrappeds des clubs | ||||
| * ``templates/wrapped/{bde_id}/wrapped_view_user.html``: le template pour les wrappeds des | ||||
|   utilisateur·ice·s | ||||
|  | ||||
| Il est conseillé de suivre la même arborescence pour les fichiers statics (fonts personnalisées, | ||||
| images, css, etc). De même, il est conseillé de créé un fichier | ||||
| ``templates/wrapped/{bde_id}/wrapped_base.html`` et d'étendre cette template. | ||||
| @@ -43,6 +43,11 @@ On a ensuite besoin de définir nos propres scopes afin d'avoir des permissions | ||||
|        'SCOPES_BACKEND_CLASS': 'permission.scopes.PermissionScopes', | ||||
|        'OAUTH2_VALIDATOR_CLASS': "permission.scopes.PermissionOAuth2Validator", | ||||
|        'REFRESH_TOKEN_EXPIRE_SECONDS': timedelta(days=14), | ||||
|        'PKCE_REQUIRED': False, | ||||
|        'OIDC_ENABLED': True, | ||||
|        'OIDC_RSA_PRIVATE_KEY': | ||||
|            os.getenv('OIDC_RSA_PRIVATE_KEY', '/var/secrets/oidc.key'), | ||||
|        'SCOPES': { 'openid': "OpenID Connect scope" }, | ||||
|    } | ||||
|  | ||||
| Cela a pour effet d'avoir des scopes sous la forme ``PERMISSION_CLUB``, | ||||
| @@ -57,6 +62,14 @@ On ajoute enfin les routes dans ``urls.py`` : | ||||
|         path('o/', include('oauth2_provider.urls', namespace='oauth2_provider')) | ||||
|     ) | ||||
|  | ||||
| Enfin pour utiliser OIDC, il faut générer une clé privé que l'on va, par défaut, | ||||
| mettre dans `/var/secrets/oidc.key` : | ||||
|  | ||||
| .. code:: bash | ||||
|  | ||||
|    cd /var/secrets/ | ||||
|    openssl genrsa -out oidc.key 4096 | ||||
|  | ||||
| L'OAuth2 est désormais prêt à être utilisé. | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -227,6 +227,22 @@ En production, ce fichier contient : | ||||
|    ) | ||||
|  | ||||
|  | ||||
| Génération d'une clé privé pour OIDC | ||||
| ------------------------------------ | ||||
|  | ||||
| Pour pouvoir proposer le service de connexion Openid Connect (OIDC) par OAuth2, il y a | ||||
| besoin d'une clé privé. Par défaut, elle est cherché dans le fichier `/var/secrets/oidc.key` | ||||
| (sinon, il faut modifier l'emplacement dans les fichiers de configurations). | ||||
|  | ||||
| Pour générer la clé, il faut aller dans le dossier `/var/secrets` (à créer, si nécessaire) puis | ||||
| utiliser la commande de génération : | ||||
|  | ||||
| .. code:: bash | ||||
|  | ||||
|    cd /var/secrets | ||||
|    openssl genrsa -out oidc.key 4096 | ||||
|  | ||||
|  | ||||
| Configuration des tâches récurrentes | ||||
| ------------------------------------ | ||||
|  | ||||
|   | ||||
| @@ -7,7 +7,7 @@ msgid "" | ||||
| msgstr "" | ||||
| "Project-Id-Version: \n" | ||||
| "Report-Msgid-Bugs-To: \n" | ||||
| "POT-Creation-Date: 2025-02-25 13:47+0100\n" | ||||
| "POT-Creation-Date: 2025-03-25 11:16+0100\n" | ||||
| "PO-Revision-Date: 2022-04-11 22:05+0200\n" | ||||
| "Last-Translator: bleizi <bleizi@crans.org>\n" | ||||
| "Language-Team: French <http://translate.ynerant.fr/projects/nk20/nk20/fr/>\n" | ||||
| @@ -25,7 +25,7 @@ msgid "This opener already exists" | ||||
| msgstr "Cette amitié existe déjà" | ||||
|  | ||||
| #: apps/activity/apps.py:10 apps/activity/models.py:129 | ||||
| #: apps/activity/models.py:169 apps/activity/models.py:323 | ||||
| #: apps/activity/models.py:169 apps/activity/models.py:328 | ||||
| msgid "activity" | ||||
| msgstr "activité" | ||||
|  | ||||
| @@ -37,24 +37,24 @@ msgstr "La note du club est inactive." | ||||
| msgid "The end date must be after the start date." | ||||
| msgstr "La date de fin doit être après celle de début." | ||||
|  | ||||
| #: apps/activity/forms.py:83 apps/activity/models.py:271 | ||||
| #: apps/activity/forms.py:83 apps/activity/models.py:276 | ||||
| msgid "You can't invite someone once the activity is started." | ||||
| msgstr "" | ||||
| "Vous ne pouvez pas inviter quelqu'un une fois que l'activité a démarré." | ||||
|  | ||||
| #: apps/activity/forms.py:86 apps/activity/models.py:274 | ||||
| #: apps/activity/forms.py:86 apps/activity/models.py:279 | ||||
| msgid "This activity is not validated yet." | ||||
| msgstr "Cette activité n'est pas encore validée." | ||||
|  | ||||
| #: apps/activity/forms.py:96 apps/activity/models.py:282 | ||||
| #: apps/activity/forms.py:96 apps/activity/models.py:287 | ||||
| msgid "This person has been already invited 5 times this year." | ||||
| msgstr "Cette personne a déjà été invitée 5 fois cette année." | ||||
|  | ||||
| #: apps/activity/forms.py:100 apps/activity/models.py:286 | ||||
| #: apps/activity/forms.py:100 apps/activity/models.py:291 | ||||
| msgid "This person is already invited." | ||||
| msgstr "Cette personne est déjà invitée." | ||||
|  | ||||
| #: apps/activity/forms.py:104 apps/activity/models.py:290 | ||||
| #: apps/activity/forms.py:104 apps/activity/models.py:295 | ||||
| msgid "You can't invite more than 3 people to this activity." | ||||
| msgstr "Vous ne pouvez pas inviter plus de 3 personnes à cette activité." | ||||
|  | ||||
| @@ -228,32 +228,36 @@ msgstr "nom de famille" | ||||
| msgid "first name" | ||||
| msgstr "prénom" | ||||
|  | ||||
| #: apps/activity/models.py:254 | ||||
| #: apps/activity/models.py:252 | ||||
| msgid "school" | ||||
| msgstr "école" | ||||
|  | ||||
| #: apps/activity/models.py:259 | ||||
| msgid "inviter" | ||||
| msgstr "hôte" | ||||
|  | ||||
| #: apps/activity/models.py:258 | ||||
| #: apps/activity/models.py:263 | ||||
| msgid "guest" | ||||
| msgstr "invité·e" | ||||
|  | ||||
| #: apps/activity/models.py:259 | ||||
| #: apps/activity/models.py:264 | ||||
| msgid "guests" | ||||
| msgstr "invité·e·s" | ||||
|  | ||||
| #: apps/activity/models.py:312 | ||||
| #: apps/activity/models.py:317 | ||||
| msgid "Invitation" | ||||
| msgstr "Invitation" | ||||
|  | ||||
| #: apps/activity/models.py:330 apps/activity/models.py:334 | ||||
| #: apps/activity/models.py:335 apps/activity/models.py:339 | ||||
| msgid "Opener" | ||||
| msgstr "Ouvreur⋅se" | ||||
|  | ||||
| #: apps/activity/models.py:335 | ||||
| #: apps/activity/models.py:340 | ||||
| #: apps/activity/templates/activity/activity_detail.html:16 | ||||
| msgid "Openers" | ||||
| msgstr "Ouvreur⋅ses" | ||||
|  | ||||
| #: apps/activity/models.py:339 | ||||
| #: apps/activity/models.py:344 | ||||
| #, fuzzy, python-brace-format | ||||
| #| msgid "Entry for {note} to the activity {activity}" | ||||
| msgid "{opener} is opener of activity {acivity}" | ||||
| @@ -463,25 +467,25 @@ msgstr "Détails de l'activité" | ||||
| msgid "Update activity" | ||||
| msgstr "Modifier l'activité" | ||||
|  | ||||
| #: apps/activity/views.py:177 | ||||
| #: apps/activity/views.py:178 | ||||
| msgid "Invite guest to the activity \"{}\"" | ||||
| msgstr "Invitation pour l'activité « {} »" | ||||
|  | ||||
| #: apps/activity/views.py:217 | ||||
| #: apps/activity/views.py:218 | ||||
| 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:220 | ||||
| #: apps/activity/views.py:221 | ||||
| msgid "This activity does not support activity entries." | ||||
| msgstr "Cette activité ne requiert pas d'entrées." | ||||
|  | ||||
| #: apps/activity/views.py:223 | ||||
| #: apps/activity/views.py:224 | ||||
| msgid "This activity is closed." | ||||
| msgstr "Cette activité est fermée." | ||||
|  | ||||
| #: apps/activity/views.py:328 | ||||
| #: apps/activity/views.py:329 | ||||
| msgid "Entry for activity \"{}\"" | ||||
| msgstr "Entrées pour l'activité « {} »" | ||||
|  | ||||
| @@ -865,7 +869,7 @@ msgstr "Taille maximale : 2 Mo" | ||||
| msgid "This image cannot be loaded." | ||||
| msgstr "Cette image ne peut pas être chargée." | ||||
|  | ||||
| #: apps/member/forms.py:154 apps/member/views.py:103 | ||||
| #: apps/member/forms.py:154 apps/member/views.py:117 | ||||
| #: apps/registration/forms.py:33 apps/registration/views.py:282 | ||||
| msgid "An alias with a similar name already exists." | ||||
| msgstr "Un alias avec un nom similaire existe déjà." | ||||
| @@ -1194,11 +1198,11 @@ msgstr "Adhésion de {user} pour le club {club}" | ||||
| 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:388 apps/member/views.py:745 | ||||
| #: apps/member/models.py:388 apps/member/views.py:759 | ||||
| msgid "User is already a member of the club" | ||||
| msgstr "L'utilisateur·rice est déjà membre du club" | ||||
|  | ||||
| #: apps/member/models.py:400 apps/member/views.py:754 | ||||
| #: apps/member/models.py:400 apps/member/views.py:768 | ||||
| msgid "User is not a member of the parent club" | ||||
| msgstr "L'utilisateur·rice n'est pas membre du club parent" | ||||
|  | ||||
| @@ -1251,7 +1255,7 @@ msgid "Account #" | ||||
| msgstr "Compte n°" | ||||
|  | ||||
| #: apps/member/templates/member/base.html:48 | ||||
| #: apps/member/templates/member/base.html:62 apps/member/views.py:60 | ||||
| #: apps/member/templates/member/base.html:62 apps/member/views.py:61 | ||||
| #: apps/registration/templates/registration/future_profile_detail.html:48 | ||||
| #: apps/wei/templates/wei/weimembership_form.html:117 | ||||
| msgid "Update Profile" | ||||
| @@ -1312,8 +1316,8 @@ msgstr "" | ||||
| "seront à nouveau possible." | ||||
|  | ||||
| #: apps/member/templates/member/club_alias.html:10 | ||||
| #: apps/member/templates/member/profile_alias.html:10 apps/member/views.py:304 | ||||
| #: apps/member/views.py:545 | ||||
| #: apps/member/templates/member/profile_alias.html:10 apps/member/views.py:318 | ||||
| #: apps/member/views.py:559 | ||||
| msgid "Note aliases" | ||||
| msgstr "Alias de la note" | ||||
|  | ||||
| @@ -1505,51 +1509,51 @@ msgstr "Sauvegarder les changements" | ||||
| msgid "Registrations" | ||||
| msgstr "Inscriptions" | ||||
|  | ||||
| #: apps/member/views.py:73 apps/registration/forms.py:23 | ||||
| #: apps/member/views.py:74 apps/registration/forms.py:23 | ||||
| msgid "This address must be valid." | ||||
| msgstr "Cette adresse doit être valide." | ||||
|  | ||||
| #: apps/member/views.py:140 | ||||
| #: apps/member/views.py:154 | ||||
| msgid "Profile detail" | ||||
| msgstr "Détails de l'utilisateur⋅rice" | ||||
|  | ||||
| #: apps/member/views.py:206 | ||||
| #: apps/member/views.py:220 | ||||
| msgid "Search user" | ||||
| msgstr "Chercher un·e utilisateur·rice" | ||||
|  | ||||
| #: apps/member/views.py:258 | ||||
| #: apps/member/views.py:272 | ||||
| msgid "Note friendships" | ||||
| msgstr "Amitiés note" | ||||
|  | ||||
| #: apps/member/views.py:328 | ||||
| #: apps/member/views.py:342 | ||||
| msgid "Update note picture" | ||||
| msgstr "Modifier la photo de la note" | ||||
|  | ||||
| #: apps/member/views.py:377 | ||||
| #: apps/member/views.py:391 | ||||
| msgid "Manage auth token" | ||||
| msgstr "Gérer les jetons d'authentification" | ||||
|  | ||||
| #: apps/member/views.py:404 | ||||
| #: apps/member/views.py:418 | ||||
| msgid "Create new club" | ||||
| msgstr "Créer un nouveau club" | ||||
|  | ||||
| #: apps/member/views.py:423 | ||||
| #: apps/member/views.py:437 | ||||
| msgid "Search club" | ||||
| msgstr "Chercher un club" | ||||
|  | ||||
| #: apps/member/views.py:461 | ||||
| #: apps/member/views.py:475 | ||||
| msgid "Club detail" | ||||
| msgstr "Détails du club" | ||||
|  | ||||
| #: apps/member/views.py:573 | ||||
| #: apps/member/views.py:587 | ||||
| msgid "Update club" | ||||
| msgstr "Modifier le club" | ||||
|  | ||||
| #: apps/member/views.py:607 | ||||
| #: apps/member/views.py:621 | ||||
| msgid "Add new member to the club" | ||||
| msgstr "Ajouter un·e nouvelleau membre au club" | ||||
|  | ||||
| #: apps/member/views.py:736 apps/wei/views.py:991 | ||||
| #: apps/member/views.py:750 apps/wei/views.py:991 | ||||
| msgid "" | ||||
| "This user don't have enough money to join this club, and can't have a " | ||||
| "negative balance." | ||||
| @@ -1557,19 +1561,19 @@ msgstr "" | ||||
| "Cet⋅te utilisateur⋅rice n'a pas assez d'argent pour rejoindre ce club et ne " | ||||
| "peut pas avoir un solde négatif." | ||||
|  | ||||
| #: apps/member/views.py:758 | ||||
| #: apps/member/views.py:772 | ||||
| msgid "The membership must start after {:%m-%d-%Y}." | ||||
| msgstr "L'adhésion doit commencer après le {:%d/%m/%Y}." | ||||
|  | ||||
| #: apps/member/views.py:763 | ||||
| #: apps/member/views.py:777 | ||||
| msgid "The membership must begin before {:%m-%d-%Y}." | ||||
| msgstr "L'adhésion doit commencer avant le {:%d/%m/%Y}." | ||||
|  | ||||
| #: apps/member/views.py:913 | ||||
| #: apps/member/views.py:927 | ||||
| msgid "Manage roles of an user in the club" | ||||
| msgstr "Gérer les rôles d'un⋅e utilisateur⋅rice dans le club" | ||||
|  | ||||
| #: apps/member/views.py:938 | ||||
| #: apps/member/views.py:952 | ||||
| msgid "Members of the club" | ||||
| msgstr "Membres du club" | ||||
|  | ||||
| @@ -1989,10 +1993,6 @@ msgstr "Historique des transactions récentes" | ||||
| #: apps/note/templates/note/mails/weekly_report.txt:32 | ||||
| #: apps/registration/templates/registration/mails/email_validation_email.html:40 | ||||
| #: apps/registration/templates/registration/mails/email_validation_email.txt:16 | ||||
| #: apps/scripts/templates/scripts/horaires.html:35 | ||||
| #: apps/scripts/templates/scripts/horaires.txt:17 | ||||
| #: apps/scripts/templates/scripts/intro_mail.html:49 | ||||
| #: apps/scripts/templates/scripts/intro_mail.txt:25 | ||||
| msgid "Mail generated by the Note Kfet on the" | ||||
| msgstr "Mail généré par la Note Kfet le" | ||||
|  | ||||
| @@ -2084,7 +2084,7 @@ msgid "Button displayed" | ||||
| msgstr "Bouton affiché" | ||||
|  | ||||
| #: apps/note/templates/note/transactiontemplate_list.html:100 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:63 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:70 | ||||
| msgid "An error occured" | ||||
| msgstr "Une erreur s'est produite" | ||||
|  | ||||
| @@ -3662,6 +3662,14 @@ msgstr "soirée·s organisée·s" | ||||
| msgid "distinct members" | ||||
| msgstr "Membres distinct·e·s" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/1/wrapped_view_club.html:26 | ||||
| msgid "Infortunately, you doesn't have consumer this year" | ||||
| msgstr "Malheureusement, tu n'as pas de consommateur cette année" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/1/wrapped_view_club.html:28 | ||||
| msgid "Congratulations you are a real rat !" | ||||
| msgstr "Félicitations, tu es un vrai rat !" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/1/wrapped_view_user.html:13 | ||||
| msgid "You participate to the wei: " | ||||
| msgstr "Tu as participé au wei : " | ||||
| @@ -3699,7 +3707,15 @@ msgstr "avec" | ||||
| msgid "Your expenses to BDE: " | ||||
| msgstr "Tes dépenses au BDE : " | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:26 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:13 | ||||
| msgid "My wrapped" | ||||
| msgstr "Mes wrapped" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:22 | ||||
| msgid "Public wrapped" | ||||
| msgstr "Wrapped public" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:33 | ||||
| msgid "" | ||||
| "Do not forget to ask permission to people who are in your wrapped before to " | ||||
| "make them public" | ||||
| @@ -3707,19 +3723,19 @@ msgstr "" | ||||
| "N'oublies pas de demander la permission des personnes apparaissant dans un " | ||||
| "wrapped avant de le rendre public" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:33 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:40 | ||||
| msgid "Link copied" | ||||
| msgstr "Lien copié" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:58 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:65 | ||||
| msgid "Wrapped is private" | ||||
| msgstr "Le wrapped est privé" | ||||
|  | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:59 | ||||
| #: apps/wrapped/templates/wrapped/wrapped_list.html:66 | ||||
| msgid "Wrapped is public" | ||||
| msgstr "Le wrapped est public" | ||||
|  | ||||
| #: apps/wrapped/views.py:24 | ||||
| #: apps/wrapped/views.py:28 | ||||
| msgid "List of wrapped" | ||||
| msgstr "Liste des wrapped" | ||||
|  | ||||
|   | ||||
| @@ -27,5 +27,5 @@ MAILTO=notekfet2020@lists.crans.org | ||||
| # Vider les tokens Oauth2 | ||||
|  00  6     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py cleartokens -v 0 | ||||
| # Envoyer la liste des abonnés à la NL BDA | ||||
|  00  10     *   *   0     root   cd /var/www/note_kfet && env/bin/python manage.py extract_ml_registrations -t art | ||||
|  00  10     *   *   0     root   cd /var/www/note_kfet && env/bin/python manage.py extract_ml_registrations -t art -e "bda.ensparissaclay@gmail.com" | ||||
|   | ||||
| @@ -268,6 +268,10 @@ OAUTH2_PROVIDER = { | ||||
|     'OAUTH2_VALIDATOR_CLASS': "permission.scopes.PermissionOAuth2Validator", | ||||
|     'REFRESH_TOKEN_EXPIRE_SECONDS': timedelta(days=14), | ||||
|     'PKCE_REQUIRED': False, # PKCE (fix a breaking change of django-oauth-toolkit 2.0.0) | ||||
|     'OIDC_ENABLED': True, | ||||
|     'OIDC_RSA_PRIVATE_KEY': | ||||
|         os.getenv('OIDC_RSA_PRIVATE_KEY', '/var/secrets/oidc.key'), | ||||
|     'SCOPES': { 'openid': "OpenID Connect scope" }, | ||||
| } | ||||
|  | ||||
| # Take control on how widget templates are sourced | ||||
|   | ||||
							
								
								
									
										96
									
								
								note_kfet/static/css/custom.css
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							
							
						
						
									
										96
									
								
								note_kfet/static/css/custom.css
									
									
									
									
									
										
										
										Executable file → Normal file
									
								
							| @@ -61,20 +61,16 @@ mark { | ||||
| /* Make navbar more readable */ | ||||
| .navbar-dark .navbar-nav .nav-link { | ||||
|     color: rgba(255, 255, 255, .75); | ||||
|     text-shadow: 2px 2px 15px #ffeb40; | ||||
|     /* text-shadow: 2px 2px 15px #ffeb40; */ | ||||
| } | ||||
|  | ||||
| .navbar-brand { | ||||
|     text-shadow: 2px 2px 15px #ffeb40; | ||||
| } | ||||
| /* .navbar-brand { */ | ||||
| /*     text-shadow: 2px 2px 15px #ffeb40; */ | ||||
| /* } | ||||
|  | ||||
| /* Last BDE colors */ | ||||
| .bg-primary { | ||||
| /*    background-color: rgb(18, 67, 4) !important; */ | ||||
| /* MODE VIEUXCON=ON */ | ||||
| /*    background-color: rgb(166, 0, 2) !important; */ | ||||
|     background-color: rgb(0, 0, 0); | ||||
|     background-image: url('/static/img/rp_bg.png'); | ||||
|     background-color: rgb(102, 83, 105) !important; | ||||
| } | ||||
|  | ||||
| html { | ||||
| @@ -89,94 +85,52 @@ body { | ||||
| .btn-outline-primary:hover, | ||||
| .btn-outline-primary:not(:disabled):not(.disabled).active, | ||||
| .btn-outline-primary:not(:disabled):not(.disabled):active { | ||||
|     color:  rgb(0, 0, 0); | ||||
|     background-color: rgb(255, 0, 101); | ||||
|     border-color: rgb(255, 203, 32); | ||||
|     color: #fff; | ||||
|     background-color: rgb(102, 83, 105); | ||||
|     border-color: rgb(102, 83, 105); | ||||
| } | ||||
|  | ||||
| .btn-outline-primary { | ||||
|     color: #000; | ||||
|     background-color: #ffcb20; | ||||
|     border-color: #000; | ||||
|     color: rgb(102, 83, 105); | ||||
|     background-color: rgba(248, 249, 250, 0.9); | ||||
|     border-color: rgb(102, 83, 105); | ||||
| } | ||||
|  | ||||
| .turbolinks-progress-bar { | ||||
|     background-color: #ffffff; | ||||
|     background-color: #12432E; | ||||
| } | ||||
|  | ||||
| .btn-primary:hover, | ||||
| .btn-primary:not(:disabled):not(.disabled).active, | ||||
| .btn-primary:not(:disabled):not(.disabled):active { | ||||
|     color:  rgb(0, 0, 0); | ||||
|     background-color: rgb(255, 0, 101); | ||||
|     border-color: rgb(255, 203, 32); | ||||
|     color: #fff; | ||||
|     background-color: rgb(102, 83, 105); | ||||
|     border-color: rgb(102, 83, 105); | ||||
| } | ||||
|  | ||||
| .btn-primary { | ||||
|     color:  #ffcb20;  | ||||
|     background-color: #000000; | ||||
|     border-color: #ffcd20; | ||||
|     color: rgba(248, 249, 250, 0.9);  | ||||
|     background-color: rgb(102, 83, 105); | ||||
|     border-color: rgb(102, 83, 105); | ||||
| } | ||||
|  | ||||
| .border-primary { | ||||
|     border-color: rgb(255, 255, 255) !important;  | ||||
| } | ||||
|  | ||||
| .btn-secondary { | ||||
|     color:  #ff0065; | ||||
|     background-color: #000000; | ||||
|     border-color: #ff0065; | ||||
| } | ||||
|  | ||||
| .btn-secondary:hover, | ||||
| .btn-secondary:not(:disabled):not(.disabled).active, | ||||
| .btn-secondary:not(:disabled):not(.disabled):active { | ||||
|     color:  rgb(0, 0, 0); | ||||
|     background-color: rgb(255, 203, 32); | ||||
|     border-color: rgb(255, 0, 101); | ||||
| } | ||||
|  | ||||
| .btn-outline-dark-shiny { | ||||
|     background-color: #222; | ||||
|     border-color: #61605b; | ||||
|     color: rgba(255, 0, 101, 75%); | ||||
| } | ||||
|  | ||||
| .btn-outline-dark-shiny:hover, | ||||
| .btn-outline-dark-shiny:not(:disabled):not(.disabled).active, | ||||
| .btn-outline-dark-shiny:not(:disabled):not(.disabled):active { | ||||
|     color:  rgb(0, 0, 0); | ||||
|     background-color: rgb(255, 203, 32); | ||||
|     border-color: rgb(255, 0, 101); | ||||
| } | ||||
|  | ||||
| .btn-outline-dark { | ||||
|     background-color: #222; | ||||
|     border-color: #61605b; | ||||
|     color: rgba(255, 203, 32, 75%); | ||||
| } | ||||
|  | ||||
| .btn-outline-dark:hover, | ||||
| .btn-outline-dark:not(:disabled):not(.disabled).active, | ||||
| .btn-outline-dark:not(:disabled):not(.disabled):active { | ||||
|     color:  rgb(0, 0, 0); | ||||
|     background-color: rgb(255, 0, 101); | ||||
|     border-color: rgb(255, 203, 32); | ||||
|     border-color: rgb(115, 15, 115) !important;  | ||||
| } | ||||
|  | ||||
| a { | ||||
|     color: rgb(255, 0, 101); | ||||
|     color: rgb(102, 83, 105); | ||||
| } | ||||
|  | ||||
| a:hover { | ||||
|     color: rgb(255, 203, 32); | ||||
|     color: rgb(200, 30, 200); | ||||
| } | ||||
|  | ||||
| .form-control:focus { | ||||
|     box-shadow: 0 0 0 0.25rem rgb(255 0 101 / 50%); | ||||
|     border-color: rgb(255, 0, 101); | ||||
|     box-shadow: 0 0 0 0.25rem rgba(200, 30, 200, 0.25); | ||||
|     border-color: rgb(200, 30, 200); | ||||
| } | ||||
|  | ||||
| .btn-outline-primary.focus { | ||||
|   box-shadow: 0 0 0 0.25rem rgb(255 203 32 / 22%); | ||||
| } | ||||
|   box-shadow: 0 0 0 0.25rem rgba(200, 30, 200, 0.5); | ||||
|  | ||||
|   | ||||
| @@ -96,11 +96,13 @@ function displayStyle (note) { | ||||
|   if (!note) { return '' } | ||||
|   const balance = note.balance | ||||
|   var css = '' | ||||
|   var ms_per_year = 31536000000 // 365 * 24 * 3600 * 1000 | ||||
|   if (balance < -2000) { css += ' text-danger bg-dark' }  | ||||
|   else if (balance < -1000) { css += ' text-danger' }  | ||||
|   else if (balance < 0) { css += ' text-warning' } | ||||
|   if (!note.email_confirmed) { css += ' bg-primary' } | ||||
|   else if (!note.is_active || (note.membership && note.membership.date_end < new Date().toISOString())) { css += ' bg-info' } | ||||
|   if (((Date.now() - Date.parse(note.created_at))/ms_per_year) > 2.5) { css += ' font-weight-bold underline' } | ||||
|   return css | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -164,7 +164,12 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
|         </div> | ||||
|     </nav> | ||||
|     <div class="{% block containertype %}container{% endblock %} my-3"> | ||||
| 	<div id="messages"> | ||||
|         <div id="messages"> | ||||
| 	    {% if user.is_authenticated %} | ||||
| 	    	<div class="alert alert-info"> | ||||
| 		    Bravo pour votre diplomation les survis ! | ||||
| 		</div> | ||||
| 	    {% endif %} | ||||
|             {% if user.is_authenticated %} | ||||
|                 {% if not user|is_member:"BDE" %} | ||||
|                     <div class="alert alert-danger"> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user