mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-31 07:49:57 +01:00 
			
		
		
		
	Compare commits
	
		
			44 Commits
		
	
	
		
			07878d493d
			...
			fix_activi
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 0d0fdef363 | ||
|  | 821efbf78b | ||
|  | a209e0d366 | ||
|  | ef485e0628 | ||
|  | 1481aa0635 | ||
|  | 867bf9fd25 | ||
|  | 47fda0ea36 | ||
|  | 623290827a | ||
|  | a87ce625f3 | ||
|  | 3559787fa7 | ||
|  | bd6ed27ae5 | ||
|  | 43dc676747 | ||
|  | caaeab6b0b | ||
|  | 54ba786884 | ||
|  | 80e109114f | ||
|  | 787005e60d | ||
|  | 414e103686 | ||
|  | 942d887c2e | ||
|  | a63c34fe37 | ||
|  | 2be6133458 | ||
|  | 7975fe47a6 | ||
|  | 476fbceeea | ||
|  | 8fbaa0bdc8 | ||
|  | a0de63effd | ||
|  | 09fb1d227e | ||
|  | 2e27d4f05c | ||
|  | 5d16dc4e7d | ||
|  | 3c34033bf5 | ||
|  | 131f508433 | ||
|  | c1a353963a | ||
|  | 178ce2b579 | ||
|  | 9162319734 | ||
|  | 5d2a8e9b79 | ||
|  | 33c94d0720 | ||
|  | 5040e8e8ea | ||
|  | c5697c4cb4 | ||
|  | e188c5a153 | ||
|  | 94e1fdc93a | ||
|  | d1ef367bab | ||
|  | 0fbb19c5fd | ||
|  | 361de9f8b4 | ||
|  | e2426bd6a6 | ||
|  | 7fea619a9f | ||
|  | 7b5eefcc0a | 
| @@ -7,21 +7,6 @@ stages: | |||||||
| variables: | variables: | ||||||
|   GIT_SUBMODULE_STRATEGY: recursive |   GIT_SUBMODULE_STRATEGY: recursive | ||||||
|  |  | ||||||
| # Debian Bullseye |  | ||||||
| py39-django42: |  | ||||||
|   stage: test |  | ||||||
|   image: debian:bullseye |  | ||||||
|   before_script: |  | ||||||
|     - > |  | ||||||
|         apt-get update && |  | ||||||
|         apt-get install --no-install-recommends -y |  | ||||||
|         python3-django python3-django-crispy-forms |  | ||||||
|         python3-django-extensions python3-django-filters python3-django-polymorphic |  | ||||||
|         python3-djangorestframework python3-django-oauth-toolkit python3-psycopg2 python3-pil |  | ||||||
|         python3-babel python3-lockfile python3-pip python3-phonenumbers python3-memcache |  | ||||||
|         python3-bs4 python3-setuptools tox texlive-xetex |  | ||||||
|   script: tox -e py39-django42 |  | ||||||
|  |  | ||||||
| # Ubuntu 22.04 | # Ubuntu 22.04 | ||||||
| py310-django42: | py310-django42: | ||||||
|   stage: test |   stage: test | ||||||
| @@ -54,8 +39,6 @@ py311-django42: | |||||||
|         python3-bs4 python3-setuptools tox texlive-xetex |         python3-bs4 python3-setuptools tox texlive-xetex | ||||||
|   script: tox -e py311-django42 |   script: tox -e py311-django42 | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| linters: | linters: | ||||||
|   stage: quality-assurance |   stage: quality-assurance | ||||||
|   image: debian:bookworm |   image: debian:bookworm | ||||||
|   | |||||||
| @@ -265,12 +265,11 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView): | |||||||
|         # Keep only users that have a note |         # 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) | ||||||
|  |  | ||||||
|         # Keep only members |         # Keep only valid members | ||||||
|         note_qs = note_qs.filter( |         note_qs = note_qs.filter( | ||||||
|             note__noteuser__user__memberships__club=activity.attendees_club, |             note__noteuser__user__memberships__club=activity.attendees_club, | ||||||
|             note__noteuser__user__memberships__date_start__lte=timezone.now(), |             note__noteuser__user__memberships__date_start__lte=timezone.now(), | ||||||
|             note__noteuser__user__memberships__date_end__gte=timezone.now(), |             note__noteuser__user__memberships__date_end__gte=timezone.now()).exclude(note__inactivity_reason='forced') | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         # Filter with permission backend |         # Filter with permission backend | ||||||
|         note_qs = note_qs.filter(PermissionBackend.filter_queryset(self.request, Alias, "view")) |         note_qs = note_qs.filter(PermissionBackend.filter_queryset(self.request, Alias, "view")) | ||||||
| @@ -330,7 +329,7 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView): | |||||||
|         context["noteuser_ctype"] = ContentType.objects.get_for_model(NoteUser).pk |         context["noteuser_ctype"] = ContentType.objects.get_for_model(NoteUser).pk | ||||||
|         context["notespecial_ctype"] = ContentType.objects.get_for_model(NoteSpecial).pk |         context["notespecial_ctype"] = ContentType.objects.get_for_model(NoteSpecial).pk | ||||||
|  |  | ||||||
|         activities_open = Activity.objects.filter(open=True).filter( |         activities_open = Activity.objects.filter(open=True, activity_type__manage_entries=True).filter( | ||||||
|             PermissionBackend.filter_queryset(self.request, Activity, "view")).distinct().all() |             PermissionBackend.filter_queryset(self.request, Activity, "view")).distinct().all() | ||||||
|         context["activities_open"] = [a for a in activities_open |         context["activities_open"] = [a for a in activities_open | ||||||
|                                       if PermissionBackend.check_perm(self.request, |                                       if PermissionBackend.check_perm(self.request, | ||||||
|   | |||||||
| @@ -44,7 +44,8 @@ class ProfileForm(forms.ModelForm): | |||||||
|     """ |     """ | ||||||
|     A form for the extras field provided by the :model:`member.Profile` model. |     A form for the extras field provided by the :model:`member.Profile` model. | ||||||
|     """ |     """ | ||||||
|     report_frequency = forms.IntegerField(required=False, initial=0, label=_("Report frequency")) |     # Remove widget=forms.HiddenInput() if you want to use report frequency. | ||||||
|  |     report_frequency = forms.IntegerField(required=False, initial=0, label=_("Report frequency"), widget=forms.HiddenInput()) | ||||||
|  |  | ||||||
|     last_report = forms.DateTimeField(required=False, disabled=True, label=_("Last report date")) |     last_report = forms.DateTimeField(required=False, disabled=True, label=_("Last report date")) | ||||||
|  |  | ||||||
| @@ -76,7 +77,8 @@ class ProfileForm(forms.ModelForm): | |||||||
|     class Meta: |     class Meta: | ||||||
|         model = Profile |         model = Profile | ||||||
|         fields = '__all__' |         fields = '__all__' | ||||||
|         exclude = ('user', 'email_confirmed', 'registration_valid', ) |         # Remove ml_[asso]_registration from exclude if the concerned association uses nk20 to manage its mailing list. | ||||||
|  |         exclude = ('user', 'email_confirmed', 'registration_valid', 'ml_sport_registration', ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ImageForm(forms.Form): | class ImageForm(forms.Form): | ||||||
|   | |||||||
| @@ -31,3 +31,4 @@ class RoleAdmin(admin.ModelAdmin): | |||||||
|     Admin customisation for Role |     Admin customisation for Role | ||||||
|     """ |     """ | ||||||
|     list_display = ('name', ) |     list_display = ('name', ) | ||||||
|  |     filter_horizontal = ('permissions',) | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -300,9 +300,13 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
| #            join_bde = True | #            join_bde = True | ||||||
| #            join_kfet = True | #            join_kfet = True | ||||||
|  |  | ||||||
|         if not join_bde: |         if not (join_bde or any(b for _, b in join_clubs)): | ||||||
|             # This software belongs to the BDE. |             # This software belongs to the BDE. | ||||||
|             form.add_error('join_bde', _("You must join the BDE.")) |             form.add_error('join_bde', _("You must join a club.")) | ||||||
|  |             return super().form_invalid(form) | ||||||
|  |  | ||||||
|  |         if join_kfet and not join_bde: | ||||||
|  |             form.add_error('join_bde', _("You must also join the parent club BDE.")) | ||||||
|             return super().form_invalid(form) |             return super().form_invalid(form) | ||||||
|  |  | ||||||
|         # Calculate required registration fee |         # Calculate required registration fee | ||||||
|   | |||||||
| @@ -276,11 +276,22 @@ class WEISurveyAlgorithm2024(WEISurveyAlgorithm): | |||||||
|         surveys = list(self.get_survey_class()(r) for r in self.get_registrations())  # All surveys |         surveys = list(self.get_survey_class()(r) for r in self.get_registrations())  # All surveys | ||||||
|         surveys = [s for s in surveys if s.is_complete()]  # Don't consider invalid surveys |         surveys = [s for s in surveys if s.is_complete()]  # Don't consider invalid surveys | ||||||
|         # Don't manage hardcoded people |         # Don't manage hardcoded people | ||||||
|         surveys = [s for s in surveys if not hasattr(s.information, 'hardcoded') or not s.information.hardcoded] |         # surveys = [s for s in surveys if s.bus_id != None] | ||||||
|  |         # surveys = [s for s in surveys if not hasattr(s.information, 'hardcoded') or not s.information.hardcoded] | ||||||
|  |  | ||||||
|  |         # surveys = [s for s in surveys if s.registration.user_id in free_users] | ||||||
|  |  | ||||||
|  |         # hardcoded_first_year_mb = WEIMembership.objects.filter(bus != None,registration__first_year=True) | ||||||
|  |         # hardcoded_first_year = hardcoded_first_year_mb.values_list('user__id', 'bus__id') | ||||||
|  |  | ||||||
|  |         hardcoded_first_year_mb = WEIMembership.objects.filter(registration__first_year=True) | ||||||
|  |         hardcoded_first_year = {mb.user.id if mb.bus else None: mb.bus.id if mb.bus else None for mb in hardcoded_first_year_mb} | ||||||
|  |  | ||||||
|         # Reset previous algorithm run |         # Reset previous algorithm run | ||||||
|         for survey in surveys: |         for survey in surveys: | ||||||
|             survey.free() |             survey.free() | ||||||
|  |             if survey.registration.user_id in hardcoded_first_year.keys(): | ||||||
|  |                 survey.select_bus(hardcoded_first_year[survey.registration.user_id]) | ||||||
|             survey.save() |             survey.save() | ||||||
|  |  | ||||||
|         non_men = [s for s in surveys if s.registration.gender != 'male'] |         non_men = [s for s in surveys if s.registration.gender != 'male'] | ||||||
| @@ -291,9 +302,7 @@ class WEISurveyAlgorithm2024(WEISurveyAlgorithm): | |||||||
|         non_men_total = registrations.filter(~Q(gender='male')).count() |         non_men_total = registrations.filter(~Q(gender='male')).count() | ||||||
|         for bus in self.get_buses(): |         for bus in self.get_buses(): | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() | ||||||
|             # Remove hardcoded people |             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) | ||||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, |  | ||||||
|                                                        registration__information_json__icontains="hardcoded").count() |  | ||||||
|             quotas[bus] = 4 + int(non_men_total / registrations.count() * free_seats) |             quotas[bus] = 4 + int(non_men_total / registrations.count() * free_seats) | ||||||
|  |  | ||||||
|         tqdm_obj = None |         tqdm_obj = None | ||||||
| @@ -308,9 +317,6 @@ class WEISurveyAlgorithm2024(WEISurveyAlgorithm): | |||||||
|         for bus in self.get_buses(): |         for bus in self.get_buses(): | ||||||
|             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() |             free_seats = bus.size - WEIMembership.objects.filter(bus=bus, registration__first_year=False).count() | ||||||
|             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) |             free_seats -= sum(1 for s in non_men if s.information.selected_bus_pk == bus.pk) | ||||||
|             # Remove hardcoded people |  | ||||||
|             free_seats -= WEIMembership.objects.filter(bus=bus, registration__first_year=True, |  | ||||||
|                                                        registration__information_json__icontains="hardcoded").count() |  | ||||||
|             quotas[bus] = free_seats |             quotas[bus] = free_seats | ||||||
|  |  | ||||||
|         if display_tqdm: |         if display_tqdm: | ||||||
|   | |||||||
| @@ -1032,6 +1032,7 @@ msgid "" | |||||||
| "Register on the mailing list to stay informed of the sport events of the " | "Register on the mailing list to stay informed of the sport events of the " | ||||||
| "campus (1 mail/week)" | "campus (1 mail/week)" | ||||||
| msgstr "" | msgstr "" | ||||||
|  | "S'inscrire sur la liste de diffusion pour rester informé·e des actualités " | ||||||
| "sportives sur le campus (1 mail par semaine)" | "sportives sur le campus (1 mail par semaine)" | ||||||
|  |  | ||||||
| #: apps/member/models.py:113 | #: apps/member/models.py:113 | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ MAILTO=notekfet2020@lists.crans.org | |||||||
|  *   *     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py send_mail -c 1 -v 0 |  *   *     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py send_mail -c 1 -v 0 | ||||||
|  *   *     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py retry_deferred -c 1 -v 0 |  *   *     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py retry_deferred -c 1 -v 0 | ||||||
|  00  0     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py purge_mail_log 7 -v 0 |  00  0     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py purge_mail_log 7 -v 0 | ||||||
|  |  00  0     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py purge_mail_log -r failure 30 -v 0 | ||||||
| # Faire une sauvegarde de la base de données | # Faire une sauvegarde de la base de données | ||||||
|  00  2     *   *   *     root   cd /var/www/note_kfet && apps/scripts/shell/backup_db |  00  2     *   *   *     root   cd /var/www/note_kfet && apps/scripts/shell/backup_db | ||||||
| # Vérifier la cohérence de la base et mailer en cas de problème | # Vérifier la cohérence de la base et mailer en cas de problème | ||||||
| @@ -25,3 +26,5 @@ MAILTO=notekfet2020@lists.crans.org | |||||||
|  00  9     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py refresh_highlighted_buttons -v 0 |  00  9     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py refresh_highlighted_buttons -v 0 | ||||||
| # Vider les tokens Oauth2 | # Vider les tokens Oauth2 | ||||||
|  00  6     *   *   *     root   cd /var/www/note_kfet && env/bin/python manage.py cleartokens -v 0 |  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  8     *   *   0     root   cd /var/www/note_kfet && env/bin/python manage.py extract_ml_registrations -t art -v 0 | ||||||
| @@ -225,6 +225,7 @@ MEDIA_URL = '/media/' | |||||||
| # Use mailer in production to place emails in a queue before sending them to avoid spam | # Use mailer in production to place emails in a queue before sending them to avoid spam | ||||||
| EMAIL_BACKEND = 'mailer.backend.DbBackend' | EMAIL_BACKEND = 'mailer.backend.DbBackend' | ||||||
| MAILER_EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' | MAILER_EMAIL_BACKEND = 'django.core.mail.backends.smtp.EmailBackend' | ||||||
|  | MAILER_EMAIL_MAX_BATCH = 10 | ||||||
| EMAIL_USE_SSL = os.getenv('EMAIL_USE_SSL', False) | EMAIL_USE_SSL = os.getenv('EMAIL_USE_SSL', False) | ||||||
| EMAIL_HOST = os.getenv('EMAIL_HOST', 'smtp.example.org') | EMAIL_HOST = os.getenv('EMAIL_HOST', 'smtp.example.org') | ||||||
| EMAIL_PORT = os.getenv('EMAIL_PORT', 25) | EMAIL_PORT = os.getenv('EMAIL_PORT', 25) | ||||||
|   | |||||||
| @@ -66,13 +66,13 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-coffee"></i> {% trans 'Consumptions' %}</a> |                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-coffee"></i> {% trans 'Consumptions' %}</a> | ||||||
|                         </li> |                         </li> | ||||||
|                     {% endif %} |                     {% endif %} | ||||||
| 		    {% if request.user.is_authenticated %} |                     {% if request.user.is_authenticated %} | ||||||
|                     	<li class="nav-item"> |                         <li class="nav-item"> | ||||||
|                             {% url 'food:food_list' as url %} |                             {% url 'food:food_list' as url %} | ||||||
| 			    <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-cutlery"></i> {% trans 'Food' %}</a> |                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-cutlery"></i> {% trans 'Food' %}</a> | ||||||
|                     	</li> |                         </li> | ||||||
| 		    {% endif %}	 |                     {% endif %} | ||||||
|                     {% if user.is_authenticated and user|is_member:"Kfet" %} |                     {% if user.is_authenticated %} | ||||||
|                         <li class="nav-item"> |                         <li class="nav-item"> | ||||||
|                             {% url 'note:transfer' as url %} |                             {% url 'note:transfer' as url %} | ||||||
|                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-exchange"></i> {% trans 'Transfer' %}</a> |                             <a class="nav-link {% if request.path_info == url %}active{% endif %}" href="{{ url }}"><i class="fa fa-exchange"></i> {% trans 'Transfer' %}</a> | ||||||
| @@ -158,7 +158,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|         </div> |         </div> | ||||||
|     </nav> |     </nav> | ||||||
|     <div class="{% block containertype %}container{% endblock %} my-3"> |     <div class="{% block containertype %}container{% endblock %} my-3"> | ||||||
|         <div id="messages"> | 	<div id="messages"> | ||||||
|             {% if user.is_authenticated %} |             {% if user.is_authenticated %} | ||||||
|                 {% if not user|is_member:"BDE" %} |                 {% if not user|is_member:"BDE" %} | ||||||
|                     <div class="alert alert-danger"> |                     <div class="alert alert-danger"> | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,14 +1,14 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = | envlist = | ||||||
|     # Debian Bullseye Python |  | ||||||
|     py39-django42 |  | ||||||
|  |  | ||||||
|     # Ubuntu 22.04 Python |     # Ubuntu 22.04 Python | ||||||
|     py310-django42 |     py310-django42 | ||||||
|  |  | ||||||
|     # Debian Bookworm Python |     # Debian Bookworm Python | ||||||
|     py311-django42 |     py311-django42 | ||||||
|  |  | ||||||
|  |     # Ubuntu 24.04 Python | ||||||
|  |     py312-django42 | ||||||
|  |  | ||||||
|     linters |     linters | ||||||
| skipsdist = True | skipsdist = True | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user