mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-26 13:33:19 +01:00 
			
		
		
		
	Compare commits
	
		
			28 Commits
		
	
	
		
			delete_act
			...
			3ec9e7811d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 3ec9e7811d | ||
|  | 763535bea4 | ||
|  | df0d886db9 | ||
|  | 092cc37320 | ||
|  | 16b55e23af | ||
|  | 97621e8704 | ||
|  | cf4c23d1ac | ||
|  | d71105976f | ||
|  | 89cc03141b | ||
|  | ff812a028c | ||
|  | 5a8acbde00 | ||
|  | f60dc8cfa0 | ||
|  | 067dd6f9d1 | ||
|  | 7b1e32e514 | ||
|  | e88dbfd597 | ||
|  | 3d34270959 | ||
|  | 3bb99671ec | ||
|  | 0d69383dfd | ||
|  | 7b9ff119e8 | ||
|  | 108a56745c | ||
|  | 9643d7652b | ||
|  | e6f3084588 | ||
|  | 145e55da75 | ||
|  | d3ba95cdca | ||
|  | 8ffb0ebb56 | ||
|  | 5038af9e34 | ||
|  | 819b4214c9 | ||
|  | b8a93b0b75 | 
| @@ -38,6 +38,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
| </a> | ||||
|  | ||||
| <input id="alias" type="text" class="form-control" placeholder="Nom/note ..."> | ||||
| <button id="trigger" class="btn btn-secondary">Click me !</button> | ||||
|  | ||||
| <hr> | ||||
|  | ||||
| @@ -63,15 +64,46 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||
|         refreshBalance(); | ||||
|     } | ||||
|  | ||||
|     function process_qrcode() { | ||||
|         let name = alias_obj.val(); | ||||
|         $.get("/api/note/note?search=" + name + "&format=json").done( | ||||
|             function (res) { | ||||
|                 let note = res.results[0]; | ||||
|                 $.post("/api/activity/entry/?format=json", { | ||||
|                     csrfmiddlewaretoken: CSRF_TOKEN, | ||||
|                     activity: {{ activity.id }}, | ||||
|                     note: note.id, | ||||
|                     guest: null | ||||
|                 }).done(function () { | ||||
|                     addMsg(interpolate(gettext( | ||||
|                         "Entry made for %s whose balance is %s €"), | ||||
|                         [note.name, note.balance / 100]), "success", 4000); | ||||
|                     reloadTable(true); | ||||
|                 }).fail(function (xhr) { | ||||
|                     errMsg(xhr.responseJSON, 4000); | ||||
|                 }); | ||||
|             }).fail(function (xhr) { | ||||
|                 errMsg(xhr.responseJSON, 4000); | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     alias_obj.keyup(function(event) { | ||||
|         let code = event.originalEvent.keyCode | ||||
|         if (65 <= code <= 122 || code === 13) { | ||||
|             debounce(reloadTable)() | ||||
|         } | ||||
|         if (code === 0) | ||||
|             process_qrcode(); | ||||
|     }); | ||||
|  | ||||
|     $(document).ready(init); | ||||
|  | ||||
|     alias_obj2 = document.getElementById("alias"); | ||||
|     $("#trigger").click(function (e) { | ||||
|         addMsg("Clicked", "success", 1000); | ||||
|         alias_obj.val(alias_obj.val() + "\0"); | ||||
|         alias_obj2.dispatchEvent(new KeyboardEvent('keyup')); | ||||
|     }) | ||||
|     function init() { | ||||
|         $(".table-row").click(function (e) { | ||||
|             let target = e.target.parentElement; | ||||
|   | ||||
| @@ -63,7 +63,8 @@ class FoodListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, Li | ||||
|             valid_regex = is_regex(pattern) | ||||
|             suffix = '__iregex' if valid_regex else '__istartswith' | ||||
|             prefix = '^' if valid_regex else '' | ||||
|             qs = qs.filter(Q(**{f'name{suffix}': prefix + pattern})) | ||||
|             qs = qs.filter(Q(**{f'name{suffix}': prefix + pattern}) | ||||
|                            | Q(**{f'owner__name{suffix}': prefix + pattern})) | ||||
|         else: | ||||
|             qs = qs.none() | ||||
|         search_table = qs.filter(PermissionBackend.filter_queryset(self.request, Food, 'view')) | ||||
| @@ -168,7 +169,8 @@ class BasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|     template_name = "food/food_update.html" | ||||
|  | ||||
|     def get_sample_object(self): | ||||
|         return BasicFood( | ||||
|         # We choose a club which may work or BDE else | ||||
|         food = BasicFood( | ||||
|             name="", | ||||
|             owner_id=1, | ||||
|             expiry_date=timezone.now(), | ||||
| @@ -177,6 +179,14 @@ class BasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|             date_type='DLC', | ||||
|         ) | ||||
|  | ||||
|         for membership in self.request.user.memberships.all(): | ||||
|             club_id = membership.club.id | ||||
|             food.owner_id = club_id | ||||
|             if PermissionBackend.check_perm(self.request, "food.add_basicfood", food): | ||||
|                 return food | ||||
|  | ||||
|         return food | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         if QRCode.objects.filter(qr_code_number=self.kwargs['slug']).count() > 0: | ||||
| @@ -227,13 +237,22 @@ class TransformedFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|     template_name = "food/food_update.html" | ||||
|  | ||||
|     def get_sample_object(self): | ||||
|         return TransformedFood( | ||||
|         # We choose a club which may work or BDE else | ||||
|         food = TransformedFood( | ||||
|             name="", | ||||
|             owner_id=1, | ||||
|             expiry_date=timezone.now(), | ||||
|             is_ready=True, | ||||
|         ) | ||||
|  | ||||
|         for membership in self.request.user.memberships.all(): | ||||
|             club_id = membership.club.id | ||||
|             food.owner_id = club_id | ||||
|             if PermissionBackend.check_perm(self.request, "food.add_transformedfood", food): | ||||
|                 return food | ||||
|  | ||||
|         return food | ||||
|  | ||||
|     @transaction.atomic | ||||
|     def form_valid(self, form): | ||||
|         form.instance.expiry_date = timezone.now() + timedelta(days=3) | ||||
| @@ -245,10 +264,10 @@ class TransformedFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): | ||||
|         return reverse_lazy('food:transformedfood_view', kwargs={"pk": self.object.pk}) | ||||
|  | ||||
|  | ||||
| MAX_FORMS = 10 | ||||
| MAX_FORMS = 100 | ||||
|  | ||||
|  | ||||
| class ManageIngredientsView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): | ||||
| class ManageIngredientsView(LoginRequiredMixin, UpdateView): | ||||
|     """ | ||||
|     A view to manage ingredient for a transformed food | ||||
|     """ | ||||
| @@ -279,6 +298,14 @@ class ManageIngredientsView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView | ||||
|                     ingredient.end_of_life = _('Fully used in {meal}'.format( | ||||
|                         meal=self.object.name)) | ||||
|                     ingredient.save() | ||||
|         # We recalculate new expiry date and allergens | ||||
|         self.object.expiry_date = self.object.creation_date + self.object.shelf_life | ||||
|         self.object.allergens.clear() | ||||
|  | ||||
|         for ingredient in self.object.ingredients.iterator(): | ||||
|             if not (ingredient.polymorphic_ctype.model == 'basicfood' and ingredient.date_type == 'DDM'): | ||||
|                 self.object.expiry_date = min(self.object.expiry_date, ingredient.expiry_date) | ||||
|             self.object.allergens.set(self.object.allergens.union(ingredient.allergens.all())) | ||||
|  | ||||
|         self.object.save(old_ingredients=old_ingredients, old_allergens=old_allergens) | ||||
|         return HttpResponseRedirect(self.get_success_url()) | ||||
|   | ||||
							
								
								
									
										46
									
								
								apps/member/migrations/0014_create_bda.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								apps/member/migrations/0014_create_bda.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| from django.db import migrations | ||||
|  | ||||
| def create_bda(apps, schema_editor): | ||||
|     """ | ||||
|     The club BDA is now pre-injected. | ||||
|     """ | ||||
|     Club = apps.get_model("member", "club") | ||||
|     NoteClub = apps.get_model("note", "noteclub") | ||||
|     Alias = apps.get_model("note", "alias") | ||||
|     ContentType = apps.get_model('contenttypes', 'ContentType') | ||||
|     polymorphic_ctype_id = ContentType.objects.get_for_model(NoteClub).id | ||||
|      | ||||
|     Club.objects.get_or_create( | ||||
|         id=10, | ||||
|         name="BDA", | ||||
|         email="bda.ensparissaclay@gmail.com", | ||||
|         require_memberships=True, | ||||
|         membership_fee_paid=750, | ||||
|         membership_fee_unpaid=750, | ||||
|         membership_duration=396, | ||||
|         membership_start="2024-08-01", | ||||
|         membership_end="2025-09-30", | ||||
|     ) | ||||
|     NoteClub.objects.get_or_create( | ||||
|         id=1937, | ||||
|         club_id=10, | ||||
|         polymorphic_ctype_id=polymorphic_ctype_id, | ||||
|     ) | ||||
|     Alias.objects.get_or_create( | ||||
|         id=1937, | ||||
|         note_id=1937, | ||||
|         name="BDA", | ||||
|         normalized_name="bda", | ||||
|     ) | ||||
|  | ||||
|  | ||||
| class Migration(migrations.Migration): | ||||
|  | ||||
|     dependencies = [ | ||||
|         ('member', '0013_auto_20240801_1436'), | ||||
|     ] | ||||
|      | ||||
|     operations = [ | ||||
|         migrations.RunPython(create_bda), | ||||
|     ] | ||||
|  | ||||
| @@ -60,7 +60,10 @@ | ||||
| {% if user_object.pk == user.pk %} | ||||
|     <div class="text-center"> | ||||
|         <a class="small badge badge-secondary" href="{% url 'member:auth_token' %}"> | ||||
|             <i class="fa fa-cogs"></i>{% trans 'API token' %} | ||||
|             <i class="fa fa-cogs"></i> {% trans 'API token' %} | ||||
|         </a> | ||||
|         <a class="small badge badge-secondary" href="{% url 'member:qr_code' user_object.pk %}"> | ||||
|             <i class="fa fa-qrcode"></i> {% trans 'QR Code' %} | ||||
|         </a> | ||||
|     </div> | ||||
| {% endif %} | ||||
|   | ||||
							
								
								
									
										36
									
								
								apps/member/templates/member/qr_code.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								apps/member/templates/member/qr_code.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| {% extends "base.html" %} | ||||
| {% comment %} | ||||
| SPDX-License-Identifier: GPL-3.0-or-later | ||||
| {% endcomment %} | ||||
| {% load i18n %} | ||||
|  | ||||
| {% block content %} | ||||
| <div class="card bg-light"> | ||||
|   	<h3 class="card-header text-center"> | ||||
| 		{% trans "QR Code for" %} {{ user_object.username }} ({{ user_object.first_name }} {{user_object.last_name }}) | ||||
|   	</h3> | ||||
|   	<div class="text-center" id="qrcode"> | ||||
|   	</div> | ||||
| </div> | ||||
|  | ||||
|  | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extrajavascript %} | ||||
| <script src="https://cdnjs.cloudflare.com/ajax/libs/qrcodejs/1.0.0/qrcode.min.js" integrity="sha512-CNgIRecGo7nphbeZ04Sc13ka07paqdeTu0WR1IM4kNcpmBAUSHSQX0FslNhTDadL4O5SAGapGt4FodqL8My0mA==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> | ||||
| <script> | ||||
| 	var qrc = new QRCode(document.getElementById("qrcode"), { | ||||
| 		text: "{{ user_object.pk }}\0", | ||||
| 		width: 1024, | ||||
| 		height: 1024 | ||||
| 	}); | ||||
| </script> | ||||
| {% endblock %} | ||||
|  | ||||
| {% block extracss %} | ||||
| <style> | ||||
| img { | ||||
|     width: 100% | ||||
| } | ||||
| </style> | ||||
| {% endblock %} | ||||
| @@ -25,4 +25,5 @@ urlpatterns = [ | ||||
|     path('user/<int:pk>/aliases/', views.ProfileAliasView.as_view(), name="user_alias"), | ||||
|     path('user/<int:pk>/trust', views.ProfileTrustView.as_view(), name="user_trust"), | ||||
|     path('manage-auth-token/', views.ManageAuthTokens.as_view(), name='auth_token'), | ||||
|     path('user/<int:pk>/qr_code/', views.QRCodeView.as_view(), name='qr_code'), | ||||
| ] | ||||
|   | ||||
| @@ -402,6 +402,14 @@ class ManageAuthTokens(LoginRequiredMixin, TemplateView): | ||||
|         context['token'] = Token.objects.get_or_create(user=self.request.user)[0] | ||||
|         return context | ||||
|  | ||||
| class QRCodeView(LoginRequiredMixin, DetailView): | ||||
|     """ | ||||
|     Affiche le QR Code | ||||
|     """ | ||||
|     model = User | ||||
|     context_object_name = "user_object" | ||||
|     template_name = "member/qr_code.html" | ||||
|     extra_context = {"title": _("QR Code")} | ||||
|  | ||||
| # ******************************* # | ||||
| #              CLUB               # | ||||
|   | ||||
| @@ -4091,8 +4091,8 @@ | ||||
|                 158, | ||||
|                 159, | ||||
|                 160, | ||||
| 		212, | ||||
| 		222 | ||||
|                 212, | ||||
|                 222 | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
| @@ -4133,14 +4133,14 @@ | ||||
|                 50, | ||||
|                 141, | ||||
|                 169, | ||||
| 		217, | ||||
| 		218, | ||||
| 		219, | ||||
| 		220, | ||||
| 		221, | ||||
| 		247, | ||||
| 		258, | ||||
| 		259 | ||||
|                 217, | ||||
|                 218, | ||||
|                 219, | ||||
|                 220, | ||||
|                 221, | ||||
|                 247, | ||||
|                 258, | ||||
|                 259 | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
| @@ -4152,8 +4152,8 @@ | ||||
|             "name": "Pr\u00e9sident\u22c5e de club", | ||||
|             "permissions": [ | ||||
|                 62, | ||||
|                 142, | ||||
|                 135 | ||||
|                 135, | ||||
|                 142 | ||||
|             ] | ||||
|         } | ||||
|     }, | ||||
| @@ -4538,8 +4538,8 @@ | ||||
|             "name": "GC anti-VSS", | ||||
|             "permissions": [ | ||||
|                 42, | ||||
| 		135, | ||||
| 		150, | ||||
|                 135, | ||||
|                 150, | ||||
|                 163, | ||||
|                 164 | ||||
|             ] | ||||
| @@ -4555,10 +4555,137 @@ | ||||
|                 137, | ||||
|                 211, | ||||
|                 212, | ||||
| 		213, | ||||
| 		214, | ||||
| 		215, | ||||
| 		216 | ||||
|                 213, | ||||
|                 214, | ||||
|                 215, | ||||
|                 216 | ||||
|             ] | ||||
|         } | ||||
|     },   | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 23, | ||||
|             "fields": { | ||||
|             "for_club": 2, | ||||
|             "name": "Darbonne", | ||||
|             "permissions": [ | ||||
|                 30, | ||||
|                 31, | ||||
|                 32 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 24, | ||||
|             "fields": { | ||||
|             "for_club": null, | ||||
|             "name": "Staffeur⋅euse (S&L,Respo Tech,...)", | ||||
|             "permissions": [] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 25, | ||||
|             "fields": { | ||||
|             "for_club": null, | ||||
|             "name": "Référent⋅e Bus", | ||||
|             "permissions": [ | ||||
|                 22, | ||||
|                 84, | ||||
|                 115, | ||||
|                 117, | ||||
|                 118, | ||||
|                 119, | ||||
|                 120, | ||||
|                 121, | ||||
|                 122 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 28, | ||||
|             "fields": { | ||||
|             "for_club": 10, | ||||
|             "name": "Trésorièr⸱e BDA", | ||||
|             "permissions": [ | ||||
|                 55, | ||||
|                 56, | ||||
|                 57, | ||||
|                 58, | ||||
|                 135, | ||||
|                 143, | ||||
|                 176, | ||||
|                 177, | ||||
|                 178, | ||||
|                 243, | ||||
|                 260, | ||||
|                 261, | ||||
|                 262, | ||||
|                 263, | ||||
|                 264, | ||||
|                 265, | ||||
|                 266, | ||||
|                 267, | ||||
|                 268, | ||||
|                 269 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 30, | ||||
|             "fields": { | ||||
|             "for_club": 10, | ||||
|             "name": "Respo sorties", | ||||
|             "permissions": [ | ||||
|                 49,  | ||||
|                 62,  | ||||
|                 141,  | ||||
|                 241,  | ||||
|                 242,  | ||||
|                 243 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 31, | ||||
|             "fields": { | ||||
|             "for_club": 1, | ||||
|             "name": "Respo comm", | ||||
|             "permissions": [ | ||||
|                 135, | ||||
|                 244 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 32, | ||||
|             "fields": { | ||||
|             "for_club": 10, | ||||
|             "name": "Respo comm Art", | ||||
|             "permissions": [ | ||||
|                 135, | ||||
|                 245 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
|     { | ||||
|         "model": "permission.role", | ||||
|         "pk": 33, | ||||
|             "fields": { | ||||
|             "for_club": 10, | ||||
|             "name": "Respo Jam", | ||||
|             "permissions": [ | ||||
|                 247,  | ||||
|                 250,  | ||||
|                 251,  | ||||
|                 252,  | ||||
|                 253,  | ||||
|                 254 | ||||
|             ] | ||||
|         } | ||||
|     },  | ||||
| @@ -4596,5 +4723,15 @@ | ||||
|         "model": "wei.weirole", | ||||
|         "pk": 18, | ||||
|         "fields": {} | ||||
|     }, | ||||
|     { | ||||
|         "model": "wei.weirole", | ||||
|         "pk": 24, | ||||
|         "fields": {} | ||||
|     }, | ||||
|     { | ||||
|         "model": "wei.weirole", | ||||
|         "pk": 25, | ||||
|         "fields": {} | ||||
|     } | ||||
| ] | ||||
|   | ||||
| @@ -1,8 +1,10 @@ | ||||
| # Copyright (C) 2018-2025 by BDE ENS Paris-Saclay | ||||
| # SPDX-License-Identifier: GPL-3.0-or-later | ||||
|  | ||||
| from oauth2_provider.oauth2_validators import OAuth2Validator | ||||
| from oauth2_provider.scopes import BaseScopes | ||||
| from member.models import Club | ||||
| from note.models import Alias | ||||
| from note_kfet.middlewares import get_current_request | ||||
|  | ||||
| from .backends import PermissionBackend | ||||
| @@ -17,25 +19,46 @@ class PermissionScopes(BaseScopes): | ||||
|     """ | ||||
|  | ||||
|     def get_all_scopes(self): | ||||
|         return {f"{p.id}_{club.id}": f"{p.description} (club {club.name})" | ||||
|                 for p in Permission.objects.all() for club in Club.objects.all()} | ||||
|         scopes = {f"{p.id}_{club.id}": f"{p.description} (club {club.name})" | ||||
|                   for p in Permission.objects.all() for club in Club.objects.all()} | ||||
|         scopes['openid'] = "OpenID Connect" | ||||
|         return scopes | ||||
|  | ||||
|     def get_available_scopes(self, application=None, request=None, *args, **kwargs): | ||||
|         if not application: | ||||
|             return [] | ||||
|         return [f"{p.id}_{p.membership.club.id}" | ||||
|                 for t in Permission.PERMISSION_TYPES | ||||
|                 for p in PermissionBackend.get_raw_permissions(get_current_request(), t[0])] | ||||
|         scopes = [f"{p.id}_{p.membership.club.id}" | ||||
|                   for t in Permission.PERMISSION_TYPES | ||||
|                   for p in PermissionBackend.get_raw_permissions(get_current_request(), t[0])] | ||||
|         scopes.append('openid') | ||||
|         return scopes | ||||
|  | ||||
|     def get_default_scopes(self, application=None, request=None, *args, **kwargs): | ||||
|         if not application: | ||||
|             return [] | ||||
|         return [f"{p.id}_{p.membership.club.id}" | ||||
|                 for p in PermissionBackend.get_raw_permissions(get_current_request(), 'view')] | ||||
|         scopes = [f"{p.id}_{p.membership.club.id}" | ||||
|                   for p in PermissionBackend.get_raw_permissions(get_current_request(), 'view')] | ||||
|         scopes.append('openid') | ||||
|         return scopes | ||||
|  | ||||
|  | ||||
| class PermissionOAuth2Validator(OAuth2Validator): | ||||
|     oidc_claim_scope = None  # fix breaking change of django-oauth-toolkit 2.0.0 | ||||
|     oidc_claim_scope = OAuth2Validator.oidc_claim_scope | ||||
|     oidc_claim_scope.update({"name": 'openid', | ||||
|                              "normalized_name": 'openid', | ||||
|                              "email": 'openid', | ||||
|                              }) | ||||
|  | ||||
|     def get_additional_claims(self, request): | ||||
|         return { | ||||
|             "name": request.user.username, | ||||
|             "normalized_name": Alias.normalize(request.user.username), | ||||
|             "email": request.user.email, | ||||
|         } | ||||
|  | ||||
|     def get_discovery_claims(self, request): | ||||
|         claims = super().get_discovery_claims(self) | ||||
|         return claims + ["name", "normalized_name", "email"] | ||||
|  | ||||
|     def validate_scopes(self, client_id, scopes, client, request, *args, **kwargs): | ||||
|         """ | ||||
| @@ -54,6 +77,8 @@ class PermissionOAuth2Validator(OAuth2Validator): | ||||
|                 if scope in scopes: | ||||
|                     valid_scopes.add(scope) | ||||
|  | ||||
|         request.scopes = valid_scopes | ||||
|         if 'openid' in scopes: | ||||
|             valid_scopes.add('openid') | ||||
|  | ||||
|         request.scopes = valid_scopes | ||||
|         return valid_scopes | ||||
|   | ||||
| @@ -19,6 +19,7 @@ EXCLUDED = [ | ||||
|     'oauth2_provider.accesstoken', | ||||
|     'oauth2_provider.grant', | ||||
|     'oauth2_provider.refreshtoken', | ||||
|     'oauth2_provider.idtoken', | ||||
|     'sessions.session', | ||||
| ] | ||||
|  | ||||
|   | ||||
| @@ -171,7 +171,7 @@ class ScopesView(LoginRequiredMixin, TemplateView): | ||||
|             available_scopes = scopes.get_available_scopes(app) | ||||
|             context["scopes"][app] = OrderedDict() | ||||
|             items = [(k, v) for (k, v) in all_scopes.items() if k in available_scopes] | ||||
|             items.sort(key=lambda x: (int(x[0].split("_")[1]), int(x[0].split("_")[0]))) | ||||
|             # items.sort(key=lambda x: (int(x[0].split("_")[1]), int(x[0].split("_")[0]))) | ||||
|             for k, v in items: | ||||
|                 context["scopes"][app][k] = v | ||||
|  | ||||
|   | ||||
| @@ -136,7 +136,7 @@ de diffusion utiles. | ||||
|    Faîtes attention, donc où la sortie est stockée. | ||||
|  | ||||
|  | ||||
| Il prend 2 options : | ||||
| Il prend 4 options : | ||||
|  | ||||
| * ``--type``, qui prend en argument ``members`` (défaut), ``clubs``, ``events``, ``art``, | ||||
|   ``sport``, qui permet respectivement de sortir la liste des adresses mails des adhérent⋅es | ||||
| @@ -149,7 +149,10 @@ Il prend 2 options : | ||||
|   pour la ML Adhérents, pour exporter les mails des adhérents au BDE pendant n'importe  | ||||
|   laquelle des ``n+1`` dernières années.  | ||||
|  | ||||
| Le script sort sur la sortie standard la liste des adresses mails à inscrire. | ||||
| * ``--email``, qui prend en argument une chaine de caractère contenant une adresse email. | ||||
|    | ||||
| Si aucun email n'est renseigné, le script sort sur la sortie standard la liste des adresses mails à inscrire. | ||||
| Dans le cas contraire, la liste est envoyée à l'adresse passée en argument. | ||||
|  | ||||
| Attention : il y a parfois certains cas particuliers à prendre en compte, il n'est | ||||
| malheureusement pas aussi simple que de simplement supposer que ces listes sont exhaustives. | ||||
|   | ||||
		Reference in New Issue
	
	Block a user