mirror of
				https://gitlab.crans.org/bde/nk20
				synced 2025-10-25 06:13:07 +02:00 
			
		
		
		
	Compare commits
	
		
			75 Commits
		
	
	
		
			faster_ci
			...
			43d214b982
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 43d214b982 | |||
| b93e4a8d11 | |||
| b9a9704061 | |||
| fee52f326a | |||
| 317966d5c1 | |||
| 9f0a22d3d1 | |||
| a5ecdd100c | |||
| f60691846b | |||
| d5ecb72a71 | |||
| 8cf9dfb9b9 | |||
| c3ab61bd04 | |||
| 0b4b6dcb3e | |||
| 0d5f6c0332 | |||
| 7b28938cde | |||
| 35ffb36fbd | |||
|  | c4c4e9594f | ||
|  | 4166823d55 | ||
|  | dc0f3dbcef | ||
|  | b3abe9ab18 | ||
|  | 27f23b48b6 | ||
|  | 67e170d4a6 | ||
|  | 8f895dc4d7 | ||
|  | 1187577728 | ||
|  | 8a58af3b31 | ||
|  | 0c23625147 | ||
|  | 21219b9c62 | ||
|  | 5ab8beecef | ||
|  | 1ca5133026 | ||
|  | 93bc6bb245 | ||
|  | 952c4383e7 | ||
| 15dd2b8f0c | |||
| c540b6334c | |||
| 0b93968b9e | |||
| 97375ef6c0 | |||
| 36cfcd533f | |||
| 21dbc53615 | |||
| e6f10ebdac | |||
| 47968844ce | |||
| a435460e29 | |||
| b7c4360108 | |||
|  | 8d8c417c50 | ||
| 2b189af25b | |||
| 5a07c8a94f | |||
| 6cc1857eb6 | |||
| 601534d610 | |||
| c271593839 | |||
| f351794aa0 | |||
| 2793fee58c | |||
| 7a715df121 | |||
| 9308878054 | |||
| b5ccf5b800 | |||
| 5e63254439 | |||
| da96506218 | |||
| b4714b896a | |||
| cdb2647a4d | |||
| cc12e3ec63 | |||
| be168c5ada | |||
| b46ae6f856 | |||
| ec0bcbf015 | |||
| 81303b8ef8 | |||
| 910b98fefc | |||
| 5a7a219ba8 | |||
| 116451603c | |||
| b2437ef9b5 | |||
| d8c9618772 | |||
| c825dee95a | |||
| 73d27e820b | |||
| 40e1b42078 | |||
| 72806f0ace | |||
| b244e01231 | |||
| 76d1784aea | |||
| 56c5fa4057 | |||
| b5ef937a03 | |||
| e95a8b6e18 | |||
| 635adf1360 | 
| @@ -1,3 +0,0 @@ | |||||||
| skip_list: |  | ||||||
|   - command-instead-of-shell  # Use shell only when shell functionality is required |  | ||||||
|   - experimental  # all rules tagged as experimental |  | ||||||
| @@ -10,7 +10,6 @@ DJANGO_SECRET_KEY=CHANGE_ME | |||||||
| DJANGO_SETTINGS_MODULE=note_kfet.settings | DJANGO_SETTINGS_MODULE=note_kfet.settings | ||||||
| CONTACT_EMAIL=tresorerie.bde@localhost | CONTACT_EMAIL=tresorerie.bde@localhost | ||||||
| NOTE_URL=localhost | NOTE_URL=localhost | ||||||
| DOMAIN=localhost |  | ||||||
|  |  | ||||||
| # Config for mails. Only used in production | # Config for mails. Only used in production | ||||||
| NOTE_MAIL=notekfet@localhost | NOTE_MAIL=notekfet@localhost | ||||||
|   | |||||||
| @@ -10,22 +10,50 @@ variables: | |||||||
| # Debian Buster | # Debian Buster | ||||||
| py37-django22: | py37-django22: | ||||||
|   stage: test |   stage: test | ||||||
|   image: otthorn/nk20_ci_37 |   image: debian:buster-backports | ||||||
|  |   before_script: | ||||||
|  |     - > | ||||||
|  |         apt-get update && | ||||||
|  |         apt-get install --no-install-recommends -t buster-backports -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 py37-django22 |   script: tox -e py37-django22 | ||||||
|  |  | ||||||
| # Ubuntu 20.04 | # Ubuntu 20.04 | ||||||
| py38-django22: | py38-django22: | ||||||
|   stage: test |   stage: test | ||||||
|   image: otthorn/nk20_ci_38 |   image: ubuntu:20.04 | ||||||
|  |   before_script: | ||||||
|  |     # Fix tzdata prompt | ||||||
|  |     - ln -sf /usr/share/zoneinfo/Europe/Paris /etc/localtime && echo Europe/Paris > /etc/timezone | ||||||
|  |     - > | ||||||
|  |         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 py38-django22 |   script: tox -e py38-django22 | ||||||
|  |  | ||||||
| # Debian Bullseye | # Debian Bullseye | ||||||
| py39-django22: | py39-django22: | ||||||
|   stage: test |   stage: test | ||||||
|   image: otthorn/nk20_ci_39 |   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-django22 |   script: tox -e py39-django22 | ||||||
|  |  | ||||||
| # Tox linter |  | ||||||
| linters: | linters: | ||||||
|   stage: quality-assurance |   stage: quality-assurance | ||||||
|   image: debian:buster-backports |   image: debian:buster-backports | ||||||
| @@ -36,20 +64,6 @@ linters: | |||||||
|   # Be nice to new contributors, but please use `tox` |   # Be nice to new contributors, but please use `tox` | ||||||
|   allow_failure: true |   allow_failure: true | ||||||
|  |  | ||||||
| # Ansible linter |  | ||||||
| ansible-linter: |  | ||||||
|   stage: quality-assurance |  | ||||||
|   image: otthorn/nk20_ci_ansiblelint |  | ||||||
|   script: ansible-lint ansible/ |  | ||||||
|  |  | ||||||
| # Docker linter |  | ||||||
| docker-linter: |  | ||||||
|   stage: quality-assurance |  | ||||||
|   image: hadolint/hadolint |  | ||||||
|   script: |  | ||||||
|     - hadolint -c .hadolint Dockerfile |  | ||||||
|     - hadolint -c .hadolint docker_ci/Dockerfile.* |  | ||||||
|  |  | ||||||
| # Compile documentation | # Compile documentation | ||||||
| documentation: | documentation: | ||||||
|   stage: docs |   stage: docs | ||||||
|   | |||||||
| @@ -1,4 +0,0 @@ | |||||||
| ignored: |  | ||||||
|   - DL3008  # Do not force to pin version in apt (Debian) |  | ||||||
|   - DL3013  # Do not force to pin version in pip (PyPI) |  | ||||||
|   - DL3018  # Do not force to pin version in apk (Alpine) |  | ||||||
| @@ -279,7 +279,8 @@ Le cahier des charges initial est disponible [sur le Wiki Crans](https://wiki.cr | |||||||
| La documentation des classes et fonctions est directement dans le code et est explorable à partir de la partie documentation de l'interface d'administration de Django. | La documentation des classes et fonctions est directement dans le code et est explorable à partir de la partie documentation de l'interface d'administration de Django. | ||||||
| **Commentez votre code !** | **Commentez votre code !** | ||||||
|  |  | ||||||
| La documentation plus haut niveau sur le développement est disponible sur [le Wiki associé au dépôt Git](https://gitlab.crans.org/bde/nk20/-/wikis/home). | La documentation plus haut niveau sur le développement et sur l'utilisation | ||||||
|  | est disponible sur <https://note.crans.org/doc> et également dans le dossier `docs`. | ||||||
|  |  | ||||||
| ## FAQ | ## FAQ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'activity.apps.ActivityConfig' | default_app_config = 'activity.apps.ActivityConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib import admin | from django.contrib import admin | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from rest_framework import serializers | from rest_framework import serializers | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .views import ActivityTypeViewSet, ActivityViewSet, EntryViewSet, GuestViewSet | from .views import ActivityTypeViewSet, ActivityViewSet, EntryViewSet, GuestViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from api.viewsets import ReadProtectedModelViewSet | from api.viewsets import ReadProtectedModelViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import os | import os | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.html import format_html | from django.utils.html import format_html | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from hashlib import md5 | from hashlib import md5 | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'api.apps.APIConfig' | default_app_config = 'api.apps.APIConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,13 +1,17 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  |  | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| from rest_framework.serializers import ModelSerializer | from django.utils import timezone | ||||||
|  | from rest_framework import serializers | ||||||
|  | from member.api.serializers import ProfileSerializer, MembershipSerializer | ||||||
|  | from note.api.serializers import NoteSerializer | ||||||
|  | from note.models import Alias | ||||||
|  |  | ||||||
|  |  | ||||||
| class UserSerializer(ModelSerializer): | class UserSerializer(serializers.ModelSerializer): | ||||||
|     """ |     """ | ||||||
|     REST API Serializer for Users. |     REST API Serializer for Users. | ||||||
|     The djangorestframework plugin will analyse the model `User` and parse all fields in the API. |     The djangorestframework plugin will analyse the model `User` and parse all fields in the API. | ||||||
| @@ -22,7 +26,7 @@ class UserSerializer(ModelSerializer): | |||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class ContentTypeSerializer(ModelSerializer): | class ContentTypeSerializer(serializers.ModelSerializer): | ||||||
|     """ |     """ | ||||||
|     REST API Serializer for Users. |     REST API Serializer for Users. | ||||||
|     The djangorestframework plugin will analyse the model `User` and parse all fields in the API. |     The djangorestframework plugin will analyse the model `User` and parse all fields in the API. | ||||||
| @@ -31,3 +35,42 @@ class ContentTypeSerializer(ModelSerializer): | |||||||
|     class Meta: |     class Meta: | ||||||
|         model = ContentType |         model = ContentType | ||||||
|         fields = '__all__' |         fields = '__all__' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class OAuthSerializer(serializers.ModelSerializer): | ||||||
|  |     """ | ||||||
|  |     Informations that are transmitted by OAuth. | ||||||
|  |     For now, this includes user, profile and valid memberships. | ||||||
|  |     This should be better managed later. | ||||||
|  |     """ | ||||||
|  |     normalized_name = serializers.SerializerMethodField() | ||||||
|  |  | ||||||
|  |     profile = ProfileSerializer() | ||||||
|  |  | ||||||
|  |     note = NoteSerializer() | ||||||
|  |  | ||||||
|  |     memberships = serializers.SerializerMethodField() | ||||||
|  |  | ||||||
|  |     def get_normalized_name(self, obj): | ||||||
|  |         return Alias.normalize(obj.username) | ||||||
|  |  | ||||||
|  |     def get_memberships(self, obj): | ||||||
|  |         return serializers.ListSerializer(child=MembershipSerializer()).to_representation( | ||||||
|  |             obj.memberships.filter(date_start__lte=timezone.now(), date_end__gte=timezone.now())) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         model = User | ||||||
|  |         fields = ( | ||||||
|  |             'id', | ||||||
|  |             'username', | ||||||
|  |             'normalized_name', | ||||||
|  |             'first_name', | ||||||
|  |             'last_name', | ||||||
|  |             'email', | ||||||
|  |             'is_superuser', | ||||||
|  |             'is_active', | ||||||
|  |             'is_staff', | ||||||
|  |             'profile', | ||||||
|  |             'note', | ||||||
|  |             'memberships', | ||||||
|  |         ) | ||||||
|   | |||||||
| @@ -1,8 +1,9 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import json | import json | ||||||
| from datetime import datetime, date | from datetime import datetime, date | ||||||
|  | from decimal import Decimal | ||||||
| from urllib.parse import quote_plus | from urllib.parse import quote_plus | ||||||
| from warnings import warn | from warnings import warn | ||||||
|  |  | ||||||
| @@ -152,6 +153,8 @@ class TestAPI(TestCase): | |||||||
|                 value = value.isoformat() |                 value = value.isoformat() | ||||||
|             elif isinstance(value, ImageFieldFile): |             elif isinstance(value, ImageFieldFile): | ||||||
|                 value = value.name |                 value = value.name | ||||||
|  |             elif isinstance(value, Decimal): | ||||||
|  |                 value = str(value) | ||||||
|             query = json.dumps({field.name: value}) |             query = json.dumps({field.name: value}) | ||||||
|  |  | ||||||
|             # Create sample permission |             # Create sample permission | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.conf.urls import url, include | from django.conf.urls import url, include | ||||||
| from rest_framework import routers | from rest_framework import routers | ||||||
|  |  | ||||||
|  | from .views import UserInformationView | ||||||
| from .viewsets import ContentTypeViewSet, UserViewSet | from .viewsets import ContentTypeViewSet, UserViewSet | ||||||
|  |  | ||||||
| # Routers provide an easy way of automatically determining the URL conf. | # Routers provide an easy way of automatically determining the URL conf. | ||||||
| @@ -47,5 +48,6 @@ app_name = 'api' | |||||||
| # Additionally, we include login URLs for the browsable API. | # Additionally, we include login URLs for the browsable API. | ||||||
| urlpatterns = [ | urlpatterns = [ | ||||||
|     url('^', include(router.urls)), |     url('^', include(router.urls)), | ||||||
|  |     url('^me/', UserInformationView.as_view()), | ||||||
|     url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), |     url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')), | ||||||
| ] | ] | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								apps/api/views.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								apps/api/views.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  | from django.contrib.auth.models import User | ||||||
|  | from rest_framework.generics import RetrieveAPIView | ||||||
|  |  | ||||||
|  | from .serializers import OAuthSerializer | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class UserInformationView(RetrieveAPIView): | ||||||
|  |     """ | ||||||
|  |     These fields are give to OAuth authenticators. | ||||||
|  |     """ | ||||||
|  |     serializer_class = OAuthSerializer | ||||||
|  |  | ||||||
|  |     def get_queryset(self): | ||||||
|  |         return User.objects.filter(pk=self.request.user.pk) | ||||||
|  |  | ||||||
|  |     def get_object(self): | ||||||
|  |         return self.request.user | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'logs.apps.LogsConfig' | default_app_config = 'logs.apps.LogsConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from rest_framework import serializers | from rest_framework import serializers | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .views import ChangelogViewSet | from .views import ChangelogViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django_filters.rest_framework import DjangoFilterBackend | from django_filters.rest_framework import DjangoFilterBackend | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib.contenttypes.models import ContentType | from django.contrib.contenttypes.models import ContentType | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'member.apps.MemberConfig' | default_app_config = 'member.apps.MemberConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib import admin | from django.contrib import admin | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from rest_framework import serializers | from rest_framework import serializers | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .views import ProfileViewSet, ClubViewSet, MembershipViewSet | from .views import ProfileViewSet, ClubViewSet, MembershipViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django_filters.rest_framework import DjangoFilterBackend | from django_filters.rest_framework import DjangoFilterBackend | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
							
								
								
									
										17
									
								
								apps/member/auth.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								apps/member/auth.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  | from cas_server.auth import DjangoAuthUser  # pragma: no cover | ||||||
|  | from note.models import Alias | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class CustomAuthUser(DjangoAuthUser):  # pragma: no cover | ||||||
|  |     """ | ||||||
|  |     Override Django Auth User model to define a custom Matrix username. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def attributs(self): | ||||||
|  |         d = super().attributs() | ||||||
|  |         if self.user: | ||||||
|  |             d["normalized_name"] = Alias.normalize(self.user.username) | ||||||
|  |         return d | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import io | import io | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import hashlib | import hashlib | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								apps/member/migrations/0007_auto_20210313_1235.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								apps/member/migrations/0007_auto_20210313_1235.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | # Generated by Django 2.2.19 on 2021-03-13 11:35 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('member', '0006_create_note_account_bde_membership'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='membership', | ||||||
|  |             name='roles', | ||||||
|  |             field=models.ManyToManyField(related_name='memberships', to='permission.Role', verbose_name='roles'), | ||||||
|  |         ), | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='profile', | ||||||
|  |             name='promotion', | ||||||
|  |             field=models.PositiveSmallIntegerField(default=2021, help_text='Year of entry to the school (None if not ENS student)', null=True, verbose_name='promotion'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import datetime | import datetime | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import date | from datetime import date | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import date | from datetime import date | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import hashlib | import hashlib | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import timedelta, date | from datetime import timedelta, date | ||||||
| @@ -625,9 +625,6 @@ class ClubAddMemberView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|         # Retrieve form data |         # Retrieve form data | ||||||
|         credit_type = form.cleaned_data["credit_type"] |         credit_type = form.cleaned_data["credit_type"] | ||||||
|         credit_amount = form.cleaned_data["credit_amount"] |         credit_amount = form.cleaned_data["credit_amount"] | ||||||
|         last_name = form.cleaned_data["last_name"] |  | ||||||
|         first_name = form.cleaned_data["first_name"] |  | ||||||
|         bank = form.cleaned_data["bank"] |  | ||||||
|         soge = form.cleaned_data["soge"] and not user.profile.soge and (club.name == "BDE" or club.name == "Kfet") |         soge = form.cleaned_data["soge"] and not user.profile.soge and (club.name == "BDE" or club.name == "Kfet") | ||||||
|  |  | ||||||
|         if not credit_type: |         if not credit_type: | ||||||
| @@ -674,17 +671,9 @@ class ClubAddMemberView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|                            .format(form.instance.club.membership_end)) |                            .format(form.instance.club.membership_end)) | ||||||
|             error = True |             error = True | ||||||
|  |  | ||||||
|         if credit_amount: |         if credit_amount and not SpecialTransaction.validate_payment_form(form): | ||||||
|             if not last_name or not first_name or (not bank and credit_type.special_type == "Chèque"): |             # Check that special information for payment are filled | ||||||
|                 if not last_name: |             error = True | ||||||
|                     form.add_error('last_name', _("This field is required.")) |  | ||||||
|                     error = True |  | ||||||
|                 if not first_name: |  | ||||||
|                     form.add_error('first_name', _("This field is required.")) |  | ||||||
|                     error = True |  | ||||||
|                 if not bank and credit_type.special_type == "Chèque": |  | ||||||
|                     form.add_error('bank', _("This field is required.")) |  | ||||||
|                     error = True |  | ||||||
|  |  | ||||||
|         return not error |         return not error | ||||||
|  |  | ||||||
| @@ -746,6 +735,7 @@ class ClubAddMemberView(ProtectQuerysetMixin, ProtectedCreateView): | |||||||
|             # When we renew the BDE membership, we update the profile section |             # When we renew the BDE membership, we update the profile section | ||||||
|             # that should happens at least once a year. |             # that should happens at least once a year. | ||||||
|             user.profile.section = user.profile.section_generated |             user.profile.section = user.profile.section_generated | ||||||
|  |             user.profile._force_save = True | ||||||
|             user.profile.save() |             user.profile.save() | ||||||
|  |  | ||||||
|         # Credit note before the membership is created. |         # Credit note before the membership is created. | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'note.apps.NoteConfig' | default_app_config = 'note.apps.NoteConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib import admin | from django.contrib import admin | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .views import NotePolymorphicViewSet, AliasViewSet, ConsumerViewSet, \ | from .views import NotePolymorphicViewSet, AliasViewSet, ConsumerViewSet, \ | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @@ -74,7 +74,7 @@ class AliasViewSet(ReadProtectedModelViewSet): | |||||||
|         serializer_class = self.serializer_class |         serializer_class = self.serializer_class | ||||||
|         if self.request.method in ['PUT', 'PATCH']: |         if self.request.method in ['PUT', 'PATCH']: | ||||||
|             # alias owner cannot be change once establish |             # alias owner cannot be change once establish | ||||||
|             setattr(serializer_class.Meta, 'read_only_fields', ('note',)) |             serializer_class.Meta.read_only_fields = ('note',) | ||||||
|         return serializer_class |         return serializer_class | ||||||
|  |  | ||||||
|     def destroy(self, request, *args, **kwargs): |     def destroy(self, request, *args, **kwargs): | ||||||
| @@ -82,7 +82,7 @@ class AliasViewSet(ReadProtectedModelViewSet): | |||||||
|         try: |         try: | ||||||
|             self.perform_destroy(instance) |             self.perform_destroy(instance) | ||||||
|         except ValidationError as e: |         except ValidationError as e: | ||||||
|             return Response({e.code: e.message}, status.HTTP_400_BAD_REQUEST) |             return Response({e.code: str(e)}, status.HTTP_400_BAD_REQUEST) | ||||||
|         return Response(status=status.HTTP_204_NO_CONTENT) |         return Response(status=status.HTTP_204_NO_CONTENT) | ||||||
|  |  | ||||||
|     def get_queryset(self): |     def get_queryset(self): | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										19
									
								
								apps/note/migrations/0005_auto_20210313_1235.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								apps/note/migrations/0005_auto_20210313_1235.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | |||||||
|  | # Generated by Django 2.2.19 on 2021-03-13 11:35 | ||||||
|  |  | ||||||
|  | from django.db import migrations, models | ||||||
|  | import django.db.models.deletion | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Migration(migrations.Migration): | ||||||
|  |  | ||||||
|  |     dependencies = [ | ||||||
|  |         ('note', '0004_remove_null_tag_on_charfields'), | ||||||
|  |     ] | ||||||
|  |  | ||||||
|  |     operations = [ | ||||||
|  |         migrations.AlterField( | ||||||
|  |             model_name='alias', | ||||||
|  |             name='note', | ||||||
|  |             field=models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='alias', to='note.Note'), | ||||||
|  |         ), | ||||||
|  |     ] | ||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .notes import Alias, Note, NoteClub, NoteSpecial, NoteUser | from .notes import Alias, Note, NoteClub, NoteSpecial, NoteUser | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import unicodedata | import unicodedata | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.conf.global_settings import DEFAULT_FROM_EMAIL |  | ||||||
| from django.core.exceptions import ValidationError | from django.core.exceptions import ValidationError | ||||||
| from django.core.mail import send_mail | from django.core.mail import send_mail | ||||||
| from django.core.validators import RegexValidator | from django.core.validators import RegexValidator | ||||||
| @@ -190,8 +189,8 @@ class NoteClub(Note): | |||||||
|     def send_mail_negative_balance(self): |     def send_mail_negative_balance(self): | ||||||
|         plain_text = render_to_string("note/mails/negative_balance.txt", dict(note=self)) |         plain_text = render_to_string("note/mails/negative_balance.txt", dict(note=self)) | ||||||
|         html = render_to_string("note/mails/negative_balance.html", dict(note=self)) |         html = render_to_string("note/mails/negative_balance.html", dict(note=self)) | ||||||
|         send_mail("[Note Kfet] Passage en négatif (club {})".format(self.club.name), plain_text, DEFAULT_FROM_EMAIL, |         send_mail("[Note Kfet] Passage en négatif (club {})".format(self.club.name), plain_text, | ||||||
|                   [self.club.email], html_message=html) |                   settings.DEFAULT_FROM_EMAIL, [self.club.email], html_message=html) | ||||||
|  |  | ||||||
|  |  | ||||||
| class NoteSpecial(Note): | class NoteSpecial(Note): | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.core.exceptions import ValidationError | from django.core.exceptions import ValidationError | ||||||
| @@ -333,6 +333,36 @@ class SpecialTransaction(Transaction): | |||||||
|         self.clean() |         self.clean() | ||||||
|         super().save(*args, **kwargs) |         super().save(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def validate_payment_form(form): | ||||||
|  |         """ | ||||||
|  |         Ensure that last name and first name are filled for a form that creates a SpecialTransaction, | ||||||
|  |         and check that if the user pays with a check, then the bank field is filled. | ||||||
|  |  | ||||||
|  |         Return True iff there is no error. | ||||||
|  |         Whenever there is an error, they are inserted in the form errors. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         credit_type = form.cleaned_data["credit_type"] | ||||||
|  |         last_name = form.cleaned_data["last_name"] | ||||||
|  |         first_name = form.cleaned_data["first_name"] | ||||||
|  |         bank = form.cleaned_data["bank"] | ||||||
|  |  | ||||||
|  |         error = False | ||||||
|  |  | ||||||
|  |         if not last_name or not first_name or (not bank and credit_type.special_type == "Chèque"): | ||||||
|  |             if not last_name: | ||||||
|  |                 form.add_error('last_name', _("This field is required.")) | ||||||
|  |                 error = True | ||||||
|  |             if not first_name: | ||||||
|  |                 form.add_error('first_name', _("This field is required.")) | ||||||
|  |                 error = True | ||||||
|  |             if not bank and credit_type.special_type == "Chèque": | ||||||
|  |                 form.add_error('bank', _("This field is required.")) | ||||||
|  |                 error = True | ||||||
|  |  | ||||||
|  |         return not error | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         verbose_name = _("Special transaction") |         verbose_name = _("Special transaction") | ||||||
|         verbose_name_plural = _("Special transactions") |         verbose_name_plural = _("Special transactions") | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| // Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | // Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| // SPDX-License-Identifier: GPL-3.0-or-later | // SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| // When a transaction is performed, lock the interface to prevent spam clicks. | // When a transaction is performed, lock the interface to prevent spam clicks. | ||||||
| @@ -28,7 +28,7 @@ $(document).ready(function () { | |||||||
|  |  | ||||||
|   // Switching in double consumptions mode should update the layout |   // Switching in double consumptions mode should update the layout | ||||||
|   $('#double_conso').change(function () { |   $('#double_conso').change(function () { | ||||||
|     $('#consos_list_div').removeClass('d-none') |     document.getElementById('consos_list_div').classList.remove('d-none') | ||||||
|     $('#infos_div').attr('class', 'col-sm-5 col-xl-6') |     $('#infos_div').attr('class', 'col-sm-5 col-xl-6') | ||||||
|  |  | ||||||
|     const note_list_obj = $('#note_list') |     const note_list_obj = $('#note_list') | ||||||
| @@ -37,7 +37,7 @@ $(document).ready(function () { | |||||||
|       note_list_obj.html('') |       note_list_obj.html('') | ||||||
|  |  | ||||||
|       buttons.forEach(function (button) { |       buttons.forEach(function (button) { | ||||||
|         $('#conso_button_' + button.id).click(function () { |         document.getElementById(`conso_button_${button.id}`).addEventListener('click', () => { | ||||||
|           if (LOCK) { return } |           if (LOCK) { return } | ||||||
|           removeNote(button, 'conso_button', buttons, 'consos_list')() |           removeNote(button, 'conso_button', buttons, 'consos_list')() | ||||||
|         }) |         }) | ||||||
| @@ -46,7 +46,7 @@ $(document).ready(function () { | |||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   $('#single_conso').change(function () { |   $('#single_conso').change(function () { | ||||||
|     $('#consos_list_div').addClass('d-none') |     document.getElementById('consos_list_div').classList.add('d-none') | ||||||
|     $('#infos_div').attr('class', 'col-sm-5 col-md-4') |     $('#infos_div').attr('class', 'col-sm-5 col-md-4') | ||||||
|  |  | ||||||
|     const consos_list_obj = $('#consos_list') |     const consos_list_obj = $('#consos_list') | ||||||
| @@ -68,9 +68,9 @@ $(document).ready(function () { | |||||||
|   }) |   }) | ||||||
|  |  | ||||||
|   // Ensure we begin in single consumption. Fix issue with TurboLinks and BootstrapJS |   // Ensure we begin in single consumption. Fix issue with TurboLinks and BootstrapJS | ||||||
|   $("label[for='double_conso']").removeClass('active') |   document.querySelector("label[for='double_conso']").classList.remove('active') | ||||||
|  |  | ||||||
|   $('#consume_all').click(consumeAll) |   document.getElementById("consume_all").addEventListener('click', consumeAll) | ||||||
| }) | }) | ||||||
|  |  | ||||||
| notes = [] | notes = [] | ||||||
| @@ -127,11 +127,10 @@ function addConso (dest, amount, type, category_id, category_name, template_id, | |||||||
|       html += li('conso_button_' + button.id, button.name + |       html += li('conso_button_' + button.id, button.name + | ||||||
|                 '<span class="badge badge-dark badge-pill">' + button.quantity + '</span>') |                 '<span class="badge badge-dark badge-pill">' + button.quantity + '</span>') | ||||||
|     }) |     }) | ||||||
|  |     document.getElementById(list).innerHTML = html | ||||||
|  |  | ||||||
|     $('#' + list).html(html) |     buttons.forEach((button) => { | ||||||
|  |       document.getElementById(`conso_button_${button.id}`).addEventListener('click', () => { | ||||||
|     buttons.forEach(function (button) { |  | ||||||
|       $('#conso_button_' + button.id).click(function () { |  | ||||||
|         if (LOCK) { return } |         if (LOCK) { return } | ||||||
|         removeNote(button, 'conso_button', buttons, list)() |         removeNote(button, 'conso_button', buttons, list)() | ||||||
|       }) |       }) | ||||||
| @@ -146,12 +145,13 @@ function reset () { | |||||||
|   notes_display.length = 0 |   notes_display.length = 0 | ||||||
|   notes.length = 0 |   notes.length = 0 | ||||||
|   buttons.length = 0 |   buttons.length = 0 | ||||||
|   $('#note_list').html('') |   document.getElementById('note_list').innerHTML = '' | ||||||
|   $('#consos_list').html('') |   document.getElementById('consos_list').innerHTML = '' | ||||||
|   $('#note').val('') |   document.getElementById('note').value = '' | ||||||
|   $('#note').attr('data-original-title', '').tooltip('hide') |   document.getElementById('note').dataset.originTitle = '' | ||||||
|   $('#profile_pic').attr('src', '/static/member/img/default_picture.png') |   $('#note').tooltip('hide') | ||||||
|   $('#profile_pic_link').attr('href', '#') |   document.getElementById('profile_pic').src = '/static/member/img/default_picture.png' | ||||||
|  |   document.getElementById('profile_pic_link').href = '#' | ||||||
|   refreshHistory() |   refreshHistory() | ||||||
|   refreshBalance() |   refreshBalance() | ||||||
|   LOCK = false |   LOCK = false | ||||||
| @@ -168,7 +168,7 @@ function consumeAll () { | |||||||
|   let error = false |   let error = false | ||||||
|  |  | ||||||
|   if (notes_display.length === 0) { |   if (notes_display.length === 0) { | ||||||
|     $('#note').addClass('is-invalid') |     document.getElementById('note').classList.add('is-invalid') | ||||||
|     $('#note_list').html(li('', '<strong>Ajoutez des émetteurs.</strong>', 'text-danger')) |     $('#note_list').html(li('', '<strong>Ajoutez des émetteurs.</strong>', 'text-danger')) | ||||||
|     error = true |     error = true | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -243,7 +243,7 @@ $('#btn_transfer').click(function () { | |||||||
|     error = true |     error = true | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const amount = Math.floor(100 * amount_field.val()) |   const amount = Math.round(100 * amount_field.val()) | ||||||
|   if (amount > 2147483647) { |   if (amount > 2147483647) { | ||||||
|     amount_field.addClass('is-invalid') |     amount_field.addClass('is-invalid') | ||||||
|     $('#amount-required').html('<strong>' + gettext('The amount must stay under 21,474,836.47 €.') + '</strong>') |     $('#amount-required').html('<strong>' + gettext('The amount must stay under 21,474,836.47 €.') + '</strong>') | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import html | import html | ||||||
|   | |||||||
| @@ -57,7 +57,7 @@ SPDX-License-Identifier: GPL-2.0-or-later | |||||||
|                 <ul class="list-group list-group-flush" id="source_note_list"> |                 <ul class="list-group list-group-flush" id="source_note_list"> | ||||||
|                 </ul> |                 </ul> | ||||||
|                 <div class="card-body"> |                 <div class="card-body"> | ||||||
|                     <select id="credit_type" class="custom-select d-none"> |                     <select id="credit_type" class="form-control custom-select d-none"> | ||||||
|                         {% for special_type in special_types %} |                         {% for special_type in special_types %} | ||||||
|                             <option value="{{ special_type.id }}">{{ special_type.special_type }}</option> |                             <option value="{{ special_type.id }}">{{ special_type.special_type }}</option> | ||||||
|                         {% endfor %} |                         {% endfor %} | ||||||
| @@ -84,7 +84,7 @@ SPDX-License-Identifier: GPL-2.0-or-later | |||||||
|                 <ul class="list-group list-group-flush" id="dest_note_list"> |                 <ul class="list-group list-group-flush" id="dest_note_list"> | ||||||
|                 </ul> |                 </ul> | ||||||
|                 <div class="card-body"> |                 <div class="card-body"> | ||||||
|                     <select id="debit_type" class="custom-select d-none"> |                     <select id="debit_type" class="form-control custom-select d-none"> | ||||||
|                         {% for special_type in special_types %} |                         {% for special_type in special_types %} | ||||||
|                             <option value="{{ special_type.id }}">{{ special_type.special_type }}</option> |                             <option value="{{ special_type.id }}">{{ special_type.special_type }}</option> | ||||||
|                         {% endfor %} |                         {% endfor %} | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django import template | from django import template | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django import template | from django import template | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from api.tests import TestAPI | from api.tests import TestAPI | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import json | import json | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'permission.apps.PermissionConfig' | default_app_config = 'permission.apps.PermissionConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-lateré | # SPDX-License-Identifier: GPL-3.0-or-lateré | ||||||
|  |  | ||||||
| from django.contrib import admin | from django.contrib import admin | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from rest_framework import serializers | from rest_framework import serializers | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from .views import PermissionViewSet, RoleViewSet | from .views import PermissionViewSet, RoleViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from api.viewsets import ReadOnlyProtectedModelViewSet | from api.viewsets import ReadOnlyProtectedModelViewSet | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import date | from datetime import date | ||||||
| @@ -134,8 +134,6 @@ class PermissionBackend(ModelBackend): | |||||||
|             return False |             return False | ||||||
|  |  | ||||||
|         sess = get_current_session() |         sess = get_current_session() | ||||||
|         if sess is not None and sess.session_key is None: |  | ||||||
|             return False |  | ||||||
|  |  | ||||||
|         if user_obj.is_superuser and sess.get("permission_mask", -1) >= 42: |         if user_obj.is_superuser and sess.get("permission_mask", -1) >= 42: | ||||||
|             return True |             return True | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| import sys | import sys | ||||||
| from functools import lru_cache | from functools import lru_cache | ||||||
|   | |||||||
| @@ -3511,6 +3511,7 @@ | |||||||
| 				56, | 				56, | ||||||
| 				57, | 				57, | ||||||
| 				58, | 				58, | ||||||
|  | 				135, | ||||||
| 				137, | 				137, | ||||||
| 				143, | 				143, | ||||||
| 				147, | 				147, | ||||||
| @@ -3521,8 +3522,7 @@ | |||||||
| 				176, | 				176, | ||||||
| 				177, | 				177, | ||||||
| 				180, | 				180, | ||||||
| 				181, | 				181 | ||||||
| 				182 |  | ||||||
| 			] | 			] | ||||||
| 		} | 		} | ||||||
| 	}, | 	}, | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import functools | import functools | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from rest_framework.permissions import DjangoObjectPermissions | from rest_framework.permissions import DjangoObjectPermissions | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.core.exceptions import PermissionDenied | from django.core.exceptions import PermissionDenied | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import django_tables2 as tables | import django_tables2 as tables | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib.auth.models import AnonymousUser | from django.contrib.auth.models import AnonymousUser | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import timedelta, date | from datetime import timedelta, date | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import date | from datetime import date | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from datetime import date | from datetime import date | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'registration.apps.RegistrationConfig' | default_app_config = 'registration.apps.RegistrationConfig' | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.apps import AppConfig | from django.apps import AppConfig | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django import forms | from django import forms | ||||||
| @@ -101,14 +101,14 @@ class ValidationForm(forms.Form): | |||||||
|         required=False, |         required=False, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     join_BDE = forms.BooleanField( |     join_bde = forms.BooleanField( | ||||||
|         label=_("Join BDE Club"), |         label=_("Join BDE Club"), | ||||||
|         required=False, |         required=False, | ||||||
|         initial=True, |         initial=True, | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     # The user can join the Kfet club at the inscription |     # The user can join the Kfet club at the inscription | ||||||
|     join_Kfet = forms.BooleanField( |     join_kfet = forms.BooleanField( | ||||||
|         label=_("Join Kfet Club"), |         label=_("Join Kfet Club"), | ||||||
|         required=False, |         required=False, | ||||||
|         initial=True, |         initial=True, | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| import django_tables2 as tables | import django_tables2 as tables | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
|  |  | ||||||
| from treasury.models import SogeCredit | from treasury.models import SogeCredit | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -100,13 +100,14 @@ SPDX-License-Identifier: GPL-3.0-or-later | |||||||
|             bank.attr('disabled', true); |             bank.attr('disabled', true); | ||||||
|             bank.val('Société générale'); |             bank.val('Société générale'); | ||||||
|  |  | ||||||
|             let join_BDE = $("#id_join_BDE"); |             let join_bde = $("#id_join_bde"); | ||||||
|             join_BDE.attr('disabled', true); |  | ||||||
|             join_BDE.attr('checked', 'checked'); |  | ||||||
|  |  | ||||||
|             let join_Kfet = $("#id_join_Kfet"); |             join_bde.attr('disabled', true); | ||||||
|             join_Kfet.attr('disabled', true); |             join_bde.attr('checked', 'checked'); | ||||||
|             join_Kfet.attr('checked', 'checked'); |  | ||||||
|  |             let join_kfet = $("#id_join_kfet"); | ||||||
|  |             join_kfet.attr('disabled', true); | ||||||
|  |             join_kfet.attr('checked', 'checked'); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         soge_field.change(fillFields); |         soge_field.change(fillFields); | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.contrib.auth.models import User | from django.contrib.auth.models import User | ||||||
| @@ -196,8 +196,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=False, |             join_bde=False, | ||||||
|             join_Kfet=False, |             join_kfet=False, | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue(response.context["form"].errors) |         self.assertTrue(response.context["form"].errors) | ||||||
| @@ -210,8 +210,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=False, |             join_bde=False, | ||||||
|             join_Kfet=True, |             join_kfet=True, | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue(response.context["form"].errors) |         self.assertTrue(response.context["form"].errors) | ||||||
| @@ -224,8 +224,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="J'ai pas d'argent", |             bank="J'ai pas d'argent", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=True, |             join_kfet=True, | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue(response.context["form"].errors) |         self.assertTrue(response.context["form"].errors) | ||||||
| @@ -238,8 +238,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="", |             last_name="", | ||||||
|             first_name="", |             first_name="", | ||||||
|             bank="", |             bank="", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=True, |             join_kfet=True, | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue(response.context["form"].errors) |         self.assertTrue(response.context["form"].errors) | ||||||
| @@ -255,8 +255,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=False, |             join_kfet=False, | ||||||
|         )) |         )) | ||||||
|         self.assertEqual(response.status_code, 200) |         self.assertEqual(response.status_code, 200) | ||||||
|         self.assertTrue(response.context["form"].errors) |         self.assertTrue(response.context["form"].errors) | ||||||
| @@ -281,8 +281,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=False, |             join_kfet=False, | ||||||
|         )) |         )) | ||||||
|         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) |         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) | ||||||
|         self.user.profile.refresh_from_db() |         self.user.profile.refresh_from_db() | ||||||
| @@ -317,8 +317,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=True, |             join_kfet=True, | ||||||
|         )) |         )) | ||||||
|         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) |         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) | ||||||
|         self.user.profile.refresh_from_db() |         self.user.profile.refresh_from_db() | ||||||
| @@ -353,8 +353,8 @@ class TestValidateRegistration(TestCase): | |||||||
|             last_name="TOTO", |             last_name="TOTO", | ||||||
|             first_name="Toto", |             first_name="Toto", | ||||||
|             bank="Société générale", |             bank="Société générale", | ||||||
|             join_BDE=True, |             join_bde=True, | ||||||
|             join_Kfet=True, |             join_kfet=True, | ||||||
|         )) |         )) | ||||||
|         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) |         self.assertRedirects(response, self.user.profile.get_absolute_url(), 302, 200) | ||||||
|         self.user.profile.refresh_from_db() |         self.user.profile.refresh_from_db() | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| # Copied from https://gitlab.crans.org/bombar/codeflix/-/blob/master/codeflix/codeflix/tokens.py | # Copied from https://gitlab.crans.org/bombar/codeflix/-/blob/master/codeflix/codeflix/tokens.py | ||||||
|  |  | ||||||
| @@ -9,6 +9,7 @@ class AccountActivationTokenGenerator(PasswordResetTokenGenerator): | |||||||
|     """ |     """ | ||||||
|     Create a unique token generator to confirm email addresses. |     Create a unique token generator to confirm email addresses. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def _make_hash_value(self, user, timestamp): |     def _make_hash_value(self, user, timestamp): | ||||||
|         """ |         """ | ||||||
|         Hash the user's primary key and some user state that's sure to change |         Hash the user's primary key and some user state that's sure to change | ||||||
| @@ -23,9 +24,18 @@ class AccountActivationTokenGenerator(PasswordResetTokenGenerator): | |||||||
|         """ |         """ | ||||||
|         # Truncate microseconds so that tokens are consistent even if the |         # Truncate microseconds so that tokens are consistent even if the | ||||||
|         # database doesn't support microseconds. |         # database doesn't support microseconds. | ||||||
|         login_timestamp = '' if user.last_login is None else user.last_login.replace(microsecond=0, tzinfo=None) |         login_timestamp = ( | ||||||
|         return str(user.pk) + str(user.email) + str(user.profile.email_confirmed)\ |             "" | ||||||
|                + str(login_timestamp) + str(timestamp) |             if user.last_login is None | ||||||
|  |             else user.last_login.replace(microsecond=0, tzinfo=None) | ||||||
|  |         ) | ||||||
|  |         return ( | ||||||
|  |             str(user.pk) | ||||||
|  |             + str(user.email) | ||||||
|  |             + str(user.profile.email_confirmed) | ||||||
|  |             + str(login_timestamp) | ||||||
|  |             + str(timestamp) | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| email_validation_token = AccountActivationTokenGenerator() | email_validation_token = AccountActivationTokenGenerator() | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @@ -248,9 +248,13 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|  |  | ||||||
|     @transaction.atomic |     @transaction.atomic | ||||||
|     def form_valid(self, form): |     def form_valid(self, form): | ||||||
|  |         """ | ||||||
|  |         Finally validate the registration, with creating the membership. | ||||||
|  |         """ | ||||||
|         user = self.get_object() |         user = self.get_object() | ||||||
|  |  | ||||||
|         if Alias.objects.filter(normalized_name=Alias.normalize(user.username)).exists(): |         if Alias.objects.filter(normalized_name=Alias.normalize(user.username)).exists(): | ||||||
|  |             # Don't try to hack an existing account. | ||||||
|             form.add_error(None, _("An alias with a similar name already exists.")) |             form.add_error(None, _("An alias with a similar name already exists.")) | ||||||
|             return self.form_invalid(form) |             return self.form_invalid(form) | ||||||
|  |  | ||||||
| @@ -261,35 +265,36 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|         last_name = form.cleaned_data["last_name"] |         last_name = form.cleaned_data["last_name"] | ||||||
|         first_name = form.cleaned_data["first_name"] |         first_name = form.cleaned_data["first_name"] | ||||||
|         bank = form.cleaned_data["bank"] |         bank = form.cleaned_data["bank"] | ||||||
|         join_BDE = form.cleaned_data["join_BDE"] |         join_bde = form.cleaned_data["join_bde"] | ||||||
|         join_Kfet = form.cleaned_data["join_Kfet"] |         join_kfet = form.cleaned_data["join_kfet"] | ||||||
|  |  | ||||||
|         if soge: |         if soge: | ||||||
|             # If Société Générale pays the inscription, the user joins the two clubs |             # If Société Générale pays the inscription, the user automatically joins the two clubs. | ||||||
|             join_BDE = True |             join_bde = True | ||||||
|             join_Kfet = True |             join_kfet = True | ||||||
|  |  | ||||||
|         if not join_BDE: |         if not join_bde: | ||||||
|             form.add_error('join_BDE', _("You must join the BDE.")) |             # This software belongs to the BDE. | ||||||
|  |             form.add_error('join_bde', _("You must join the BDE.")) | ||||||
|             return super().form_invalid(form) |             return super().form_invalid(form) | ||||||
|  |  | ||||||
|  |         # Calculate required registration fee | ||||||
|         fee = 0 |         fee = 0 | ||||||
|         bde = Club.objects.get(name="BDE") |         bde = Club.objects.get(name="BDE") | ||||||
|         bde_fee = bde.membership_fee_paid if user.profile.paid else bde.membership_fee_unpaid |         bde_fee = bde.membership_fee_paid if user.profile.paid else bde.membership_fee_unpaid | ||||||
|         if join_BDE: |         # This is mandatory. | ||||||
|             fee += bde_fee |         fee += bde_fee if join_bde else 0 | ||||||
|         kfet = Club.objects.get(name="Kfet") |         kfet = Club.objects.get(name="Kfet") | ||||||
|         kfet_fee = kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid |         kfet_fee = kfet.membership_fee_paid if user.profile.paid else kfet.membership_fee_unpaid | ||||||
|         if join_Kfet: |         # Add extra fee for the full membership | ||||||
|             fee += kfet_fee |         fee += kfet_fee if join_kfet else 0 | ||||||
|  |  | ||||||
|         if soge: |         # If the bank pays, then we don't credit now. Treasurers will validate the transaction | ||||||
|             # If the bank pays, then we don't credit now. Treasurers will validate the transaction |         # and credit the note later. | ||||||
|             # and credit the note later. |         credit_type = None if soge else credit_type | ||||||
|             credit_type = None |  | ||||||
|  |  | ||||||
|         if credit_type is None: |         # If the user does not select any payment method, then no credit will be performed. | ||||||
|             credit_amount = 0 |         credit_amount = 0 if credit_type is None else credit_amount | ||||||
|  |  | ||||||
|         if fee > credit_amount and not soge: |         if fee > credit_amount and not soge: | ||||||
|             # Check if the user credits enough money |             # Check if the user credits enough money | ||||||
| @@ -298,15 +303,9 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|                            .format(pretty_money(fee))) |                            .format(pretty_money(fee))) | ||||||
|             return self.form_invalid(form) |             return self.form_invalid(form) | ||||||
|  |  | ||||||
|         if credit_type is not None and credit_amount > 0: |         # Check that payment information are filled, like last name and first name | ||||||
|             if not last_name or not first_name or (not bank and credit_type.special_type == "Chèque"): |         if credit_type is not None and credit_amount > 0 and not SpecialTransaction.validate_payment_form(form): | ||||||
|                 if not last_name: |             return self.form_invalid(form) | ||||||
|                     form.add_error('last_name', _("This field is required.")) |  | ||||||
|                 if not first_name: |  | ||||||
|                     form.add_error('first_name', _("This field is required.")) |  | ||||||
|                 if not bank and credit_type.special_type == "Chèque": |  | ||||||
|                     form.add_error('bank', _("This field is required.")) |  | ||||||
|                 return self.form_invalid(form) |  | ||||||
|  |  | ||||||
|         # Save the user and finally validate the registration |         # Save the user and finally validate the registration | ||||||
|         # Saving the user creates the associated note |         # Saving the user creates the associated note | ||||||
| @@ -338,7 +337,7 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|                 valid=True, |                 valid=True, | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         if join_BDE: |         if join_bde: | ||||||
|             # Create membership for the user to the BDE starting today |             # Create membership for the user to the BDE starting today | ||||||
|             membership = Membership( |             membership = Membership( | ||||||
|                 club=bde, |                 club=bde, | ||||||
| @@ -352,7 +351,7 @@ class FutureUserDetailView(ProtectQuerysetMixin, LoginRequiredMixin, FormMixin, | |||||||
|             membership.roles.add(Role.objects.get(name="Adhérent BDE")) |             membership.roles.add(Role.objects.get(name="Adhérent BDE")) | ||||||
|             membership.save() |             membership.save() | ||||||
|  |  | ||||||
|         if join_Kfet: |         if join_kfet: | ||||||
|             # Create membership for the user to the Kfet starting today |             # Create membership for the user to the Kfet starting today | ||||||
|             membership = Membership( |             membership = Membership( | ||||||
|                 club=kfet, |                 club=kfet, | ||||||
|   | |||||||
 Submodule apps/scripts updated: 8ec7d68a16...7a022b9407
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| # Copyright (C) 2018-2020 by BDE ENS Paris-Saclay | # Copyright (C) 2018-2021 by BDE ENS Paris-Saclay | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  |  | ||||||
| default_app_config = 'treasury.apps.TreasuryConfig' | default_app_config = 'treasury.apps.TreasuryConfig' | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user