mirror of
https://gitlab.crans.org/bde/nk20
synced 2025-06-21 18:08:21 +02:00
Compare commits
26 Commits
09117da2e0
...
faster_ci
Author | SHA1 | Date | |
---|---|---|---|
3eed93e346 | |||
4da523a1ba | |||
e74ff54468 | |||
2e49c9ffbd | |||
d20a1038a8 | |||
f6b711bb1b | |||
893d87a9e1 | |||
9f3323c73e | |||
c57f81b920 | |||
0636d84286 | |||
ed06901fae | |||
28932f316b | |||
9b50ba722c | |||
3e3e61d23f | |||
1129815ca3 | |||
c13172d3ff | |||
fcc4121225 | |||
a06f355559 | |||
08df5fcccd | |||
b6c0f9758d | |||
a23093851f | |||
d803ab5ec2 | |||
d7a537b6b5 | |||
0941ee954d | |||
fd11d96d95 | |||
4bfc057454 |
3
.ansible-lint
Normal file
3
.ansible-lint
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
skip_list:
|
||||||
|
- command-instead-of-shell # Use shell only when shell functionality is required
|
||||||
|
- experimental # all rules tagged as experimental
|
@ -10,6 +10,7 @@ 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
|
||||||
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -42,13 +42,11 @@ map.json
|
|||||||
backups/
|
backups/
|
||||||
/static/
|
/static/
|
||||||
/media/
|
/media/
|
||||||
/tmp/
|
|
||||||
|
|
||||||
# Virtualenv
|
# Virtualenv
|
||||||
env/
|
env/
|
||||||
venv/
|
venv/
|
||||||
db.sqlite3
|
db.sqlite3
|
||||||
shell.nix
|
|
||||||
|
|
||||||
# ansibles customs host
|
# ansibles customs host
|
||||||
ansible/host_vars/*.yaml
|
ansible/host_vars/*.yaml
|
||||||
|
@ -7,41 +7,28 @@ stages:
|
|||||||
variables:
|
variables:
|
||||||
GIT_SUBMODULE_STRATEGY: recursive
|
GIT_SUBMODULE_STRATEGY: recursive
|
||||||
|
|
||||||
# Ubuntu 22.04
|
# Debian Buster
|
||||||
py310-django42:
|
py37-django22:
|
||||||
stage: test
|
stage: test
|
||||||
image: ubuntu:22.04
|
image: otthorn/nk20_ci_37
|
||||||
before_script:
|
script: tox -e py37-django22
|
||||||
# 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 py310-django42
|
|
||||||
|
|
||||||
# Debian Bookworm
|
# Ubuntu 20.04
|
||||||
py311-django42:
|
py38-django22:
|
||||||
stage: test
|
stage: test
|
||||||
image: debian:bookworm
|
image: otthorn/nk20_ci_38
|
||||||
before_script:
|
script: tox -e py38-django22
|
||||||
- >
|
|
||||||
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 py311-django42
|
|
||||||
|
|
||||||
|
# Debian Bullseye
|
||||||
|
py39-django22:
|
||||||
|
stage: test
|
||||||
|
image: otthorn/nk20_ci_39
|
||||||
|
script: tox -e py39-django22
|
||||||
|
|
||||||
|
# Tox linter
|
||||||
linters:
|
linters:
|
||||||
stage: quality-assurance
|
stage: quality-assurance
|
||||||
image: debian:bookworm
|
image: debian:buster-backports
|
||||||
before_script:
|
before_script:
|
||||||
- apt-get update && apt-get install -y tox
|
- apt-get update && apt-get install -y tox
|
||||||
script: tox -e linters
|
script: tox -e linters
|
||||||
@ -49,6 +36,20 @@ 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
|
||||||
|
2
.gitmodules
vendored
2
.gitmodules
vendored
@ -1,3 +1,3 @@
|
|||||||
[submodule "apps/scripts"]
|
[submodule "apps/scripts"]
|
||||||
path = apps/scripts
|
path = apps/scripts
|
||||||
url = https://gitlab.crans.org/bde/nk20-scripts
|
url = https://gitlab.crans.org/bde/nk20-scripts.git
|
||||||
|
4
.hadolint
Normal file
4
.hadolint
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
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)
|
23
README.md
23
README.md
@ -1,8 +1,8 @@
|
|||||||
# NoteKfet 2020
|
# NoteKfet 2020
|
||||||
|
|
||||||
[](https://www.gnu.org/licenses/gpl-3.0.txt)
|
[](https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
[](https://gitlab.crans.org/bde/nk20/commits/main)
|
[](https://gitlab.crans.org/bde/nk20/commits/master)
|
||||||
[](https://gitlab.crans.org/bde/nk20/commits/main)
|
[](https://gitlab.crans.org/bde/nk20/commits/master)
|
||||||
|
|
||||||
## Table des matières
|
## Table des matières
|
||||||
|
|
||||||
@ -55,16 +55,10 @@ Bien que cela permette de créer une instance sur toutes les distributions,
|
|||||||
(env)$ ./manage.py makemigrations
|
(env)$ ./manage.py makemigrations
|
||||||
(env)$ ./manage.py migrate
|
(env)$ ./manage.py migrate
|
||||||
(env)$ ./manage.py loaddata initial
|
(env)$ ./manage.py loaddata initial
|
||||||
(env)$ ./manage.py createsuperuser # Création d'un⋅e utilisateur⋅rice initial
|
(env)$ ./manage.py createsuperuser # Création d'un utilisateur initial
|
||||||
```
|
```
|
||||||
|
|
||||||
6. (Optionnel) **Création d'une clé privée OpenID Connect**
|
6. Enjoy :
|
||||||
|
|
||||||
Pour activer le support d'OpenID Connect, il faut générer une clé privée, par
|
|
||||||
exemple avec openssl (`openssl genrsa -out oidc.key 4096`), et renseigner son
|
|
||||||
emplacement dans `OIDC_RSA_PRIVATE_KEY` (par défaut `/var/secrets/oidc.key`).
|
|
||||||
|
|
||||||
7. Enjoy :
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
(env)$ ./manage.py runserver 0.0.0.0:8000
|
(env)$ ./manage.py runserver 0.0.0.0:8000
|
||||||
@ -234,12 +228,6 @@ Sinon vous pouvez suivre les étapes décrites ci-dessous.
|
|||||||
(env)$ ./manage.py check # pas de bêtise qui traine
|
(env)$ ./manage.py check # pas de bêtise qui traine
|
||||||
(env)$ ./manage.py migrate
|
(env)$ ./manage.py migrate
|
||||||
|
|
||||||
7. **Création d'une clé privée OpenID Connect**
|
|
||||||
|
|
||||||
Pour activer le support d'OpenID Connect, il faut générer une clé privée, par
|
|
||||||
exemple avec openssl (`openssl genrsa -out oidc.key 4096`), et renseigner son
|
|
||||||
emplacement dans `OIDC_RSA_PRIVATE_KEY` (par défaut `/var/secrets/oidc.key`).
|
|
||||||
|
|
||||||
7. *Enjoy \o/*
|
7. *Enjoy \o/*
|
||||||
|
|
||||||
### Installation avec Docker
|
### Installation avec Docker
|
||||||
@ -291,8 +279,7 @@ 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 et sur l'utilisation
|
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).
|
||||||
est disponible sur <https://note.crans.org/doc> et également dans le dossier `docs`.
|
|
||||||
|
|
||||||
## FAQ
|
## FAQ
|
||||||
|
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
prompt: "Password of the database (leave it blank to skip database init)"
|
prompt: "Password of the database (leave it blank to skip database init)"
|
||||||
private: yes
|
private: yes
|
||||||
vars:
|
vars:
|
||||||
mirror: eclats.crans.org
|
mirror: mirror.crans.org
|
||||||
roles:
|
roles:
|
||||||
- 1-apt-basic
|
- 1-apt-basic
|
||||||
- 2-nk20
|
- 2-nk20
|
||||||
|
6
ansible/host_vars/bde-nk20-beta.adh.crans.org.yml
Normal file
6
ansible/host_vars/bde-nk20-beta.adh.crans.org.yml
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
note:
|
||||||
|
server_name: note-beta.crans.org
|
||||||
|
git_branch: beta
|
||||||
|
cron_enabled: false
|
||||||
|
email: notekfet2020@lists.crans.org
|
@ -2,6 +2,5 @@
|
|||||||
note:
|
note:
|
||||||
server_name: note-dev.crans.org
|
server_name: note-dev.crans.org
|
||||||
git_branch: beta
|
git_branch: beta
|
||||||
serve_static: false
|
|
||||||
cron_enabled: false
|
cron_enabled: false
|
||||||
email: notekfet2020@lists.crans.org
|
email: notekfet2020@lists.crans.org
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
---
|
---
|
||||||
note:
|
note:
|
||||||
server_name: note.crans.org
|
server_name: note.crans.org
|
||||||
git_branch: main
|
git_branch: master
|
||||||
serve_static: true
|
|
||||||
cron_enabled: true
|
cron_enabled: true
|
||||||
email: notekfet2020@lists.crans.org
|
email: notekfet2020@lists.crans.org
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
[dev]
|
[dev]
|
||||||
bde-note-dev.adh.crans.org
|
bde-note-dev.adh.crans.org
|
||||||
|
bde-nk20-beta.adh.crans.org
|
||||||
|
|
||||||
[prod]
|
[prod]
|
||||||
bde-note.adh.crans.org
|
bde-note.adh.crans.org
|
||||||
|
@ -1,15 +1,14 @@
|
|||||||
---
|
---
|
||||||
- name: Add buster-backports to apt sources if needed
|
- name: Add buster-backports to apt sources
|
||||||
apt_repository:
|
apt_repository:
|
||||||
repo: deb http://{{ mirror }}/debian buster-backports main
|
repo: deb http://{{ mirror }}/debian buster-backports main
|
||||||
state: present
|
state: present
|
||||||
when:
|
when: ansible_facts['distribution'] == "Debian"
|
||||||
- ansible_distribution == "Debian"
|
|
||||||
- ansible_distribution_major_version | int == 10
|
|
||||||
|
|
||||||
- name: Install note_kfet APT dependencies
|
- name: Install note_kfet APT dependencies
|
||||||
apt:
|
apt:
|
||||||
update_cache: true
|
update_cache: true
|
||||||
|
default_release: "{{ 'buster-backports' if ansible_facts['distribution'] == 'Debian' }}"
|
||||||
install_recommends: false
|
install_recommends: false
|
||||||
name:
|
name:
|
||||||
# Common tools
|
# Common tools
|
||||||
|
@ -41,7 +41,6 @@ server {
|
|||||||
# max upload size
|
# max upload size
|
||||||
client_max_body_size 75M; # adjust to taste
|
client_max_body_size 75M; # adjust to taste
|
||||||
|
|
||||||
{% if note.serve_static %}
|
|
||||||
# Django media
|
# Django media
|
||||||
location /media {
|
location /media {
|
||||||
alias /var/www/note_kfet/media; # your Django project's media files - amend as required
|
alias /var/www/note_kfet/media; # your Django project's media files - amend as required
|
||||||
@ -51,7 +50,6 @@ server {
|
|||||||
alias /var/www/note_kfet/static; # your Django project's static files - amend as required
|
alias /var/www/note_kfet/static; # your Django project's static files - amend as required
|
||||||
}
|
}
|
||||||
|
|
||||||
{% endif %}
|
|
||||||
location /doc {
|
location /doc {
|
||||||
alias /var/www/documentation; # The documentation of the project
|
alias /var/www/documentation; # The documentation of the project
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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,11 +1,11 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
from note_kfet.admin import admin_site
|
from note_kfet.admin import admin_site
|
||||||
|
|
||||||
from .forms import GuestForm
|
from .forms import GuestForm
|
||||||
from .models import Activity, ActivityType, Entry, Guest, Opener
|
from .models import Activity, ActivityType, Entry, Guest
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Activity, site=admin_site)
|
@admin.register(Activity, site=admin_site)
|
||||||
@ -45,11 +45,3 @@ class EntryAdmin(admin.ModelAdmin):
|
|||||||
Admin customisation for Entry
|
Admin customisation for Entry
|
||||||
"""
|
"""
|
||||||
list_display = ('note', 'activity', 'time', 'guest')
|
list_display = ('note', 'activity', 'time', 'guest')
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Opener, site=admin_site)
|
|
||||||
class OpenerAdmin(admin.ModelAdmin):
|
|
||||||
"""
|
|
||||||
Admin customisation for Opener
|
|
||||||
"""
|
|
||||||
list_display = ('activity', 'opener')
|
|
||||||
|
@ -1,11 +1,9 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
from rest_framework.validators import UniqueTogetherValidator
|
|
||||||
|
|
||||||
from ..models import Activity, ActivityType, Entry, Guest, GuestTransaction, Opener
|
from ..models import Activity, ActivityType, Entry, Guest, GuestTransaction
|
||||||
|
|
||||||
|
|
||||||
class ActivityTypeSerializer(serializers.ModelSerializer):
|
class ActivityTypeSerializer(serializers.ModelSerializer):
|
||||||
@ -61,17 +59,3 @@ class GuestTransactionSerializer(serializers.ModelSerializer):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = GuestTransaction
|
model = GuestTransaction
|
||||||
fields = '__all__'
|
fields = '__all__'
|
||||||
|
|
||||||
|
|
||||||
class OpenerSerializer(serializers.ModelSerializer):
|
|
||||||
"""
|
|
||||||
REST API Serializer for Openers.
|
|
||||||
The djangorestframework plugin will analyse the model `Opener` and parse all fields in the API.
|
|
||||||
"""
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = Opener
|
|
||||||
fields = '__all__'
|
|
||||||
validators = [UniqueTogetherValidator(
|
|
||||||
queryset=Opener.objects.all(), fields=("opener", "activity"),
|
|
||||||
message=_("This opener already exists"))]
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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, OpenerViewSet
|
from .views import ActivityTypeViewSet, ActivityViewSet, EntryViewSet, GuestViewSet
|
||||||
|
|
||||||
|
|
||||||
def register_activity_urls(router, path):
|
def register_activity_urls(router, path):
|
||||||
@ -12,4 +12,3 @@ def register_activity_urls(router, path):
|
|||||||
router.register(path + '/type', ActivityTypeViewSet)
|
router.register(path + '/type', ActivityTypeViewSet)
|
||||||
router.register(path + '/guest', GuestViewSet)
|
router.register(path + '/guest', GuestViewSet)
|
||||||
router.register(path + '/entry', EntryViewSet)
|
router.register(path + '/entry', EntryViewSet)
|
||||||
router.register(path + '/opener', OpenerViewSet)
|
|
||||||
|
@ -1,15 +1,12 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
from api.filters import RegexSafeSearchFilter
|
|
||||||
from api.viewsets import ReadProtectedModelViewSet
|
from api.viewsets import ReadProtectedModelViewSet
|
||||||
from django.core.exceptions import ValidationError
|
|
||||||
from django_filters.rest_framework import DjangoFilterBackend
|
from django_filters.rest_framework import DjangoFilterBackend
|
||||||
from rest_framework.response import Response
|
from rest_framework.filters import SearchFilter
|
||||||
from rest_framework import status
|
|
||||||
|
|
||||||
from .serializers import ActivitySerializer, ActivityTypeSerializer, EntrySerializer, GuestSerializer, OpenerSerializer
|
from .serializers import ActivitySerializer, ActivityTypeSerializer, EntrySerializer, GuestSerializer
|
||||||
from ..models import Activity, ActivityType, Entry, Guest, Opener
|
from ..models import Activity, ActivityType, Entry, Guest
|
||||||
|
|
||||||
|
|
||||||
class ActivityTypeViewSet(ReadProtectedModelViewSet):
|
class ActivityTypeViewSet(ReadProtectedModelViewSet):
|
||||||
@ -32,7 +29,7 @@ class ActivityViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Activity.objects.order_by('id')
|
queryset = Activity.objects.order_by('id')
|
||||||
serializer_class = ActivitySerializer
|
serializer_class = ActivitySerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['name', 'description', 'activity_type', 'location', 'creater', 'organizer', 'attendees_club',
|
filterset_fields = ['name', 'description', 'activity_type', 'location', 'creater', 'organizer', 'attendees_club',
|
||||||
'date_start', 'date_end', 'valid', 'open', ]
|
'date_start', 'date_end', 'valid', 'open', ]
|
||||||
search_fields = ['$name', '$description', '$location', '$creater__last_name', '$creater__first_name',
|
search_fields = ['$name', '$description', '$location', '$creater__last_name', '$creater__first_name',
|
||||||
@ -50,7 +47,7 @@ class GuestViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Guest.objects.order_by('id')
|
queryset = Guest.objects.order_by('id')
|
||||||
serializer_class = GuestSerializer
|
serializer_class = GuestSerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['activity', 'activity__name', 'last_name', 'first_name', 'inviter', 'inviter__alias__name',
|
filterset_fields = ['activity', 'activity__name', 'last_name', 'first_name', 'inviter', 'inviter__alias__name',
|
||||||
'inviter__alias__normalized_name', ]
|
'inviter__alias__normalized_name', ]
|
||||||
search_fields = ['$activity__name', '$last_name', '$first_name', '$inviter__user__email', '$inviter__alias__name',
|
search_fields = ['$activity__name', '$last_name', '$first_name', '$inviter__user__email', '$inviter__alias__name',
|
||||||
@ -65,36 +62,7 @@ class EntryViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Entry.objects.order_by('id')
|
queryset = Entry.objects.order_by('id')
|
||||||
serializer_class = EntrySerializer
|
serializer_class = EntrySerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['activity', 'time', 'note', 'guest', ]
|
filterset_fields = ['activity', 'time', 'note', 'guest', ]
|
||||||
search_fields = ['$activity__name', '$note__user__email', '$note__alias__name', '$note__alias__normalized_name',
|
search_fields = ['$activity__name', '$note__user__email', '$note__alias__name', '$note__alias__normalized_name',
|
||||||
'$guest__last_name', '$guest__first_name', ]
|
'$guest__last_name', '$guest__first_name', ]
|
||||||
|
|
||||||
|
|
||||||
class OpenerViewSet(ReadProtectedModelViewSet):
|
|
||||||
"""
|
|
||||||
REST Opener View set.
|
|
||||||
The djangorestframework plugin will get all `Opener` objects, serialize it to JSON with the given serializer,
|
|
||||||
then render it on /api/activity/opener/
|
|
||||||
"""
|
|
||||||
queryset = Opener.objects
|
|
||||||
serializer_class = OpenerSerializer
|
|
||||||
filter_backends = [RegexSafeSearchFilter, DjangoFilterBackend]
|
|
||||||
search_fields = ['$opener__alias__name', '$opener__alias__normalized_name',
|
|
||||||
'$activity__name']
|
|
||||||
filterset_fields = ['opener', 'opener__noteuser__user', 'activity']
|
|
||||||
|
|
||||||
def get_serializer_class(self):
|
|
||||||
serializer_class = self.serializer_class
|
|
||||||
if self.request.method in ['PUT', 'PATCH']:
|
|
||||||
# opener-activity can't change
|
|
||||||
serializer_class.Meta.read_only_fields = ('opener', 'acitivity',)
|
|
||||||
return serializer_class
|
|
||||||
|
|
||||||
def destroy(self, request, *args, **kwargs):
|
|
||||||
instance = self.get_object()
|
|
||||||
try:
|
|
||||||
self.perform_destroy(instance)
|
|
||||||
except ValidationError as e:
|
|
||||||
return Response({e.code: str(e)}, status.HTTP_400_BAD_REQUEST)
|
|
||||||
return Response(status=status.HTTP_204_NO_CONTENT)
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
"name": "Pot",
|
"name": "Pot",
|
||||||
"manage_entries": true,
|
"manage_entries": true,
|
||||||
"can_invite": true,
|
"can_invite": true,
|
||||||
"guest_entry_fee": 1000
|
"guest_entry_fee": 500
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -28,25 +28,5 @@
|
|||||||
"can_invite": false,
|
"can_invite": false,
|
||||||
"guest_entry_fee": 0
|
"guest_entry_fee": 0
|
||||||
}
|
}
|
||||||
},
|
|
||||||
{
|
|
||||||
"model": "activity.activitytype",
|
|
||||||
"pk": 5,
|
|
||||||
"fields": {
|
|
||||||
"name": "Soir\u00e9e avec entrées",
|
|
||||||
"manage_entries": true,
|
|
||||||
"can_invite": false,
|
|
||||||
"guest_entry_fee": 0
|
|
||||||
}
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"model": "activity.activitytype",
|
|
||||||
"pk": 7,
|
|
||||||
"fields": {
|
|
||||||
"name": "Soir\u00e9e avec invitations",
|
|
||||||
"manage_entries": true,
|
|
||||||
"can_invite": true,
|
|
||||||
"guest_entry_fee": 0
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
@ -1,18 +1,17 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
from random import shuffle
|
from random import shuffle
|
||||||
|
|
||||||
from bootstrap_datepicker_plus.widgets import DateTimePickerInput
|
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from member.models import Club
|
from member.models import Club
|
||||||
from note.models import Note, NoteUser
|
from note.models import Note, NoteUser
|
||||||
from note_kfet.inputs import Autocomplete
|
from note_kfet.inputs import Autocomplete, DateTimePickerInput
|
||||||
from note_kfet.middlewares import get_current_request
|
from note_kfet.middlewares import get_current_authenticated_user
|
||||||
from permission.backends import PermissionBackend
|
from permission.backends import PermissionBackend
|
||||||
|
|
||||||
from .models import Activity, Guest
|
from .models import Activity, Guest
|
||||||
@ -25,16 +24,10 @@ class ActivityForm(forms.ModelForm):
|
|||||||
self.fields["attendees_club"].initial = Club.objects.get(name="Kfet")
|
self.fields["attendees_club"].initial = Club.objects.get(name="Kfet")
|
||||||
self.fields["attendees_club"].widget.attrs["placeholder"] = "Kfet"
|
self.fields["attendees_club"].widget.attrs["placeholder"] = "Kfet"
|
||||||
clubs = list(Club.objects.filter(PermissionBackend
|
clubs = list(Club.objects.filter(PermissionBackend
|
||||||
.filter_queryset(get_current_request(), Club, "view")).all())
|
.filter_queryset(get_current_authenticated_user(), Club, "view")).all())
|
||||||
shuffle(clubs)
|
shuffle(clubs)
|
||||||
self.fields["organizer"].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
self.fields["organizer"].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
||||||
|
|
||||||
def clean_organizer(self):
|
|
||||||
organizer = self.cleaned_data['organizer']
|
|
||||||
if not organizer.note.is_active:
|
|
||||||
self.add_error('organiser', _('The note of this club is inactive.'))
|
|
||||||
return organizer
|
|
||||||
|
|
||||||
def clean_date_end(self):
|
def clean_date_end(self):
|
||||||
date_end = self.cleaned_data["date_end"]
|
date_end = self.cleaned_data["date_end"]
|
||||||
date_start = self.cleaned_data["date_start"]
|
date_start = self.cleaned_data["date_start"]
|
||||||
@ -44,7 +37,7 @@ class ActivityForm(forms.ModelForm):
|
|||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Activity
|
model = Activity
|
||||||
exclude = ('creater', 'valid', 'open', 'opener', )
|
exclude = ('creater', 'valid', 'open', )
|
||||||
widgets = {
|
widgets = {
|
||||||
"organizer": Autocomplete(
|
"organizer": Autocomplete(
|
||||||
model=Club,
|
model=Club,
|
||||||
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-03-23 13:22
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('activity', '0002_auto_20200904_2341'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='activity',
|
|
||||||
name='description',
|
|
||||||
field=models.TextField(blank=True, default='', verbose_name='description'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,28 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-08-01 12:36
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
import django.db.models.deletion
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('note', '0006_trust'),
|
|
||||||
('activity', '0003_auto_20240323_1422'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='Opener',
|
|
||||||
fields=[
|
|
||||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
||||||
('activity', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='opener', to='activity.Activity', verbose_name='activity')),
|
|
||||||
('opener', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='activity_responsible', to='note.Note', verbose_name='opener')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'opener',
|
|
||||||
'verbose_name_plural': 'openers',
|
|
||||||
'unique_together': {('opener', 'activity')},
|
|
||||||
},
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,24 +0,0 @@
|
|||||||
# Generated by Django 4.2.15 on 2024-08-28 08:00
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
import django.db.models.deletion
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('note', '0006_trust'),
|
|
||||||
('activity', '0004_opener'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterModelOptions(
|
|
||||||
name='opener',
|
|
||||||
options={'verbose_name': 'Opener', 'verbose_name_plural': 'Openers'},
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='opener',
|
|
||||||
name='opener',
|
|
||||||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='activity_responsible', to='note.note', verbose_name='Opener'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
@ -11,7 +11,7 @@ from django.db import models, transaction
|
|||||||
from django.db.models import Q
|
from django.db.models import Q
|
||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from note.models import NoteUser, Transaction, Note
|
from note.models import NoteUser, Transaction
|
||||||
from rest_framework.exceptions import ValidationError
|
from rest_framework.exceptions import ValidationError
|
||||||
|
|
||||||
|
|
||||||
@ -66,8 +66,6 @@ class Activity(models.Model):
|
|||||||
|
|
||||||
description = models.TextField(
|
description = models.TextField(
|
||||||
verbose_name=_('description'),
|
verbose_name=_('description'),
|
||||||
blank=True,
|
|
||||||
default="",
|
|
||||||
)
|
)
|
||||||
|
|
||||||
location = models.CharField(
|
location = models.CharField(
|
||||||
@ -125,14 +123,6 @@ class Activity(models.Model):
|
|||||||
verbose_name=_('open'),
|
verbose_name=_('open'),
|
||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _("activity")
|
|
||||||
verbose_name_plural = _("activities")
|
|
||||||
unique_together = ("name", "date_start", "date_end",)
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def save(self, *args, **kwargs):
|
def save(self, *args, **kwargs):
|
||||||
"""
|
"""
|
||||||
@ -154,6 +144,14 @@ class Activity(models.Model):
|
|||||||
if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else refresh_activities()
|
if settings.DATABASES["default"]["ENGINE"] == 'django.db.backends.postgresql' else refresh_activities()
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("activity")
|
||||||
|
verbose_name_plural = _("activities")
|
||||||
|
unique_together = ("name", "date_start", "date_end",)
|
||||||
|
|
||||||
|
|
||||||
class Entry(models.Model):
|
class Entry(models.Model):
|
||||||
"""
|
"""
|
||||||
@ -254,13 +252,14 @@ class Guest(models.Model):
|
|||||||
verbose_name=_("inviter"),
|
verbose_name=_("inviter"),
|
||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
@property
|
||||||
verbose_name = _("guest")
|
def has_entry(self):
|
||||||
verbose_name_plural = _("guests")
|
try:
|
||||||
unique_together = ("activity", "last_name", "first_name", )
|
if self.entry:
|
||||||
|
return True
|
||||||
def __str__(self):
|
return False
|
||||||
return self.first_name + " " + self.last_name
|
except AttributeError:
|
||||||
|
return False
|
||||||
|
|
||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
|
def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
|
||||||
@ -291,14 +290,13 @@ class Guest(models.Model):
|
|||||||
|
|
||||||
return super().save(force_insert, force_update, using, update_fields)
|
return super().save(force_insert, force_update, using, update_fields)
|
||||||
|
|
||||||
@property
|
def __str__(self):
|
||||||
def has_entry(self):
|
return self.first_name + " " + self.last_name
|
||||||
try:
|
|
||||||
if self.entry:
|
class Meta:
|
||||||
return True
|
verbose_name = _("guest")
|
||||||
return False
|
verbose_name_plural = _("guests")
|
||||||
except AttributeError:
|
unique_together = ("activity", "last_name", "first_name", )
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class GuestTransaction(Transaction):
|
class GuestTransaction(Transaction):
|
||||||
@ -310,31 +308,3 @@ class GuestTransaction(Transaction):
|
|||||||
@property
|
@property
|
||||||
def type(self):
|
def type(self):
|
||||||
return _('Invitation')
|
return _('Invitation')
|
||||||
|
|
||||||
|
|
||||||
class Opener(models.Model):
|
|
||||||
"""
|
|
||||||
Allow the user to make activity entries without more rights
|
|
||||||
"""
|
|
||||||
activity = models.ForeignKey(
|
|
||||||
Activity,
|
|
||||||
on_delete=models.CASCADE,
|
|
||||||
related_name='opener',
|
|
||||||
verbose_name=_('activity')
|
|
||||||
)
|
|
||||||
|
|
||||||
opener = models.ForeignKey(
|
|
||||||
Note,
|
|
||||||
on_delete=models.CASCADE,
|
|
||||||
related_name='activity_responsible',
|
|
||||||
verbose_name=_('Opener')
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _("Opener")
|
|
||||||
verbose_name_plural = _("Openers")
|
|
||||||
unique_together = ("opener", "activity")
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return _("{opener} is opener of activity {acivity}").format(
|
|
||||||
opener=str(self.opener), acivity=str(self.activity))
|
|
||||||
|
@ -1,57 +0,0 @@
|
|||||||
/**
|
|
||||||
* On form submit, add a new opener
|
|
||||||
*/
|
|
||||||
function form_create_opener (e) {
|
|
||||||
// Do not submit HTML form
|
|
||||||
e.preventDefault()
|
|
||||||
|
|
||||||
// Get data and send to API
|
|
||||||
const formData = new FormData(e.target)
|
|
||||||
$.getJSON('/api/note/alias/'+formData.get('opener') + '/',
|
|
||||||
function (opener_alias) {
|
|
||||||
create_opener(formData.get('activity'), opener_alias.note)
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add an opener between an activity and a user
|
|
||||||
* @param activity:Integer activity id
|
|
||||||
* @param opener:Integer user note id
|
|
||||||
*/
|
|
||||||
function create_opener(activity, opener) {
|
|
||||||
$.post('/api/activity/opener/', {
|
|
||||||
activity: activity,
|
|
||||||
opener: opener,
|
|
||||||
csrfmiddlewaretoken: CSRF_TOKEN
|
|
||||||
}).done(function () {
|
|
||||||
// Reload tables
|
|
||||||
$('#opener_table').load(location.pathname + ' #opener_table')
|
|
||||||
addMsg(gettext('Opener successfully added'), 'success')
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* On click of "delete", delete the opener
|
|
||||||
* @param button_id:Integer Opener id to remove
|
|
||||||
*/
|
|
||||||
function delete_button (button_id) {
|
|
||||||
$.ajax({
|
|
||||||
url: '/api/activity/opener/' + button_id + '/',
|
|
||||||
method: 'DELETE',
|
|
||||||
headers: { 'X-CSRFTOKEN': CSRF_TOKEN }
|
|
||||||
}).done(function () {
|
|
||||||
addMsg(gettext('Opener successfully deleted'), 'success')
|
|
||||||
$('#opener_table').load(location.pathname + ' #opener_table')
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
$(document).ready(function () {
|
|
||||||
// Attach event
|
|
||||||
document.getElementById('form_opener').addEventListener('submit', form_create_opener)
|
|
||||||
})
|
|
@ -1,17 +1,13 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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 escape
|
from django.utils.html import format_html
|
||||||
from django.utils.safestring import mark_safe
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from note_kfet.middlewares import get_current_request
|
|
||||||
import django_tables2 as tables
|
import django_tables2 as tables
|
||||||
from django_tables2 import A
|
from django_tables2 import A
|
||||||
from permission.backends import PermissionBackend
|
|
||||||
from note.templatetags.pretty_money import pretty_money
|
from note.templatetags.pretty_money import pretty_money
|
||||||
|
|
||||||
from .models import Activity, Entry, Guest, Opener
|
from .models import Activity, Entry, Guest
|
||||||
|
|
||||||
|
|
||||||
class ActivityTable(tables.Table):
|
class ActivityTable(tables.Table):
|
||||||
@ -56,8 +52,8 @@ class GuestTable(tables.Table):
|
|||||||
def render_entry(self, record):
|
def render_entry(self, record):
|
||||||
if record.has_entry:
|
if record.has_entry:
|
||||||
return str(_("Entered on ") + str(_("{:%Y-%m-%d %H:%M:%S}").format(record.entry.time, )))
|
return str(_("Entered on ") + str(_("{:%Y-%m-%d %H:%M:%S}").format(record.entry.time, )))
|
||||||
return mark_safe('<button id="{id}" class="btn btn-danger btn-sm" onclick="remove_guest(this.id)"> '
|
return format_html('<button id="{id}" class="btn btn-danger btn-sm" onclick="remove_guest(this.id)"> '
|
||||||
'{delete_trans}</button>'.format(id=record.id, delete_trans=_("remove").capitalize()))
|
'{delete_trans}</button>'.format(id=record.id, delete_trans=_("remove").capitalize()))
|
||||||
|
|
||||||
|
|
||||||
def get_row_class(record):
|
def get_row_class(record):
|
||||||
@ -95,7 +91,7 @@ class EntryTable(tables.Table):
|
|||||||
if hasattr(record, 'username'):
|
if hasattr(record, 'username'):
|
||||||
username = record.username
|
username = record.username
|
||||||
if username != value:
|
if username != value:
|
||||||
return mark_safe(escape(value) + " <em>aka.</em> " + escape(username))
|
return format_html(value + " <em>aka.</em> " + username)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def render_balance(self, value):
|
def render_balance(self, value):
|
||||||
@ -115,34 +111,3 @@ class EntryTable(tables.Table):
|
|||||||
'data-last-name': lambda record: record.last_name,
|
'data-last-name': lambda record: record.last_name,
|
||||||
'data-first-name': lambda record: record.first_name,
|
'data-first-name': lambda record: record.first_name,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
# function delete_button(id) provided in template file
|
|
||||||
DELETE_TEMPLATE = """
|
|
||||||
<button id="{{ record.pk }}" class="btn btn-danger btn-sm" onclick="delete_button(this.id)"> {{ delete_trans }}</button>
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class OpenerTable(tables.Table):
|
|
||||||
class Meta:
|
|
||||||
attrs = {
|
|
||||||
'class': 'table table condensed table-striped',
|
|
||||||
'id': "opener_table"
|
|
||||||
}
|
|
||||||
model = Opener
|
|
||||||
fields = ("opener",)
|
|
||||||
template_name = 'django_tables2/bootstrap4.html'
|
|
||||||
|
|
||||||
show_header = False
|
|
||||||
opener = tables.Column(attrs={'td': {'class': 'text-center'}})
|
|
||||||
|
|
||||||
delete_col = tables.TemplateColumn(
|
|
||||||
template_code=DELETE_TEMPLATE,
|
|
||||||
extra_context={"delete_trans": _('Delete')},
|
|
||||||
attrs={
|
|
||||||
'td': {
|
|
||||||
'class': lambda record: 'col-sm-1'
|
|
||||||
+ (' d-none' if not PermissionBackend.check_perm(
|
|
||||||
get_current_request(), "activity.delete_opener", record)
|
|
||||||
else '')}},
|
|
||||||
verbose_name=_("Delete"),)
|
|
||||||
|
@ -4,31 +4,11 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
{% endcomment %}
|
{% endcomment %}
|
||||||
{% load i18n perms %}
|
{% load i18n perms %}
|
||||||
{% load render_table from django_tables2 %}
|
{% load render_table from django_tables2 %}
|
||||||
{% load static django_tables2 i18n %}
|
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<h1 class="text-white">{{ title }}</h1>
|
<h1 class="text-white">{{ title }}</h1>
|
||||||
{% include "activity/includes/activity_info.html" %}
|
{% include "activity/includes/activity_info.html" %}
|
||||||
|
|
||||||
{% if activity.activity_type.manage_entries and ".change__opener"|has_perm:activity %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{% trans "Openers" %}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body">
|
|
||||||
<form class="input-group" method="POST" id="form_opener">
|
|
||||||
{% csrf_token %}
|
|
||||||
<input type="hidden" name="activity" value="{{ object.pk }}">
|
|
||||||
{%include "autocomplete_model.html" %}
|
|
||||||
<div class="input-group-append">
|
|
||||||
<input type="submit" class="btn btn-success" value="{% trans "Add" %}">
|
|
||||||
</div>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
{% render_table opener %}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if guests.data %}
|
{% if guests.data %}
|
||||||
<div class="card bg-white mb-3">
|
<div class="card bg-white mb-3">
|
||||||
<h3 class="card-header text-center">
|
<h3 class="card-header text-center">
|
||||||
@ -42,8 +22,6 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extrajavascript %}
|
{% block extrajavascript %}
|
||||||
<script src="{% static "activity/js/opener.js" %}"></script>
|
|
||||||
<script src="{% static "js/autocomplete_model.js" %}"></script>
|
|
||||||
<script>
|
<script>
|
||||||
function remove_guest(guest_id) {
|
function remove_guest(guest_id) {
|
||||||
$.ajax({
|
$.ajax({
|
||||||
|
@ -63,12 +63,7 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
refreshBalance();
|
refreshBalance();
|
||||||
}
|
}
|
||||||
|
|
||||||
alias_obj.keyup(function(event) {
|
alias_obj.keyup(reloadTable);
|
||||||
let code = event.originalEvent.keyCode
|
|
||||||
if (65 <= code <= 122 || code === 13) {
|
|
||||||
debounce(reloadTable)()
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
$(document).ready(init);
|
$(document).ready(init);
|
||||||
|
|
||||||
|
@ -18,26 +18,3 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
{% block extrajavascript %}
|
|
||||||
<script>
|
|
||||||
var date_end = document.getElementById("id_date_end");
|
|
||||||
var date_start = document.getElementById("id_date_start");
|
|
||||||
|
|
||||||
function update_date_end (){
|
|
||||||
if(date_end.value=="" || date_end.value<date_start.value){
|
|
||||||
date_end.value = date_start.value;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
function update_date_start (){
|
|
||||||
if(date_start.value=="" || date_end.value<date_start.value){
|
|
||||||
date_start.value = date_end.value;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
date_start.addEventListener('focusout', update_date_end);
|
|
||||||
date_end.addEventListener('focusout', update_date_start);
|
|
||||||
|
|
||||||
</script>
|
|
||||||
{% endblock %}
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
@ -17,16 +17,14 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from django.views import View
|
from django.views import View
|
||||||
from django.views.decorators.cache import cache_page
|
from django.views.decorators.cache import cache_page
|
||||||
from django.views.generic import DetailView, TemplateView, UpdateView
|
from django.views.generic import DetailView, TemplateView, UpdateView
|
||||||
from django.views.generic.list import ListView
|
from django_tables2.views import SingleTableView
|
||||||
from django_tables2.views import MultiTableMixin, SingleTableMixin
|
|
||||||
from api.viewsets import is_regex
|
|
||||||
from note.models import Alias, NoteSpecial, NoteUser
|
from note.models import Alias, NoteSpecial, NoteUser
|
||||||
from permission.backends import PermissionBackend
|
from permission.backends import PermissionBackend
|
||||||
from permission.views import ProtectQuerysetMixin, ProtectedCreateView
|
from permission.views import ProtectQuerysetMixin, ProtectedCreateView
|
||||||
|
|
||||||
from .forms import ActivityForm, GuestForm
|
from .forms import ActivityForm, GuestForm
|
||||||
from .models import Activity, Entry, Guest, Opener
|
from .models import Activity, Entry, Guest
|
||||||
from .tables import ActivityTable, EntryTable, GuestTable, OpenerTable
|
from .tables import ActivityTable, EntryTable, GuestTable
|
||||||
|
|
||||||
|
|
||||||
class ActivityCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
class ActivityCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
||||||
@ -59,44 +57,36 @@ class ActivityCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
|||||||
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk})
|
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.object.pk})
|
||||||
|
|
||||||
|
|
||||||
class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView):
|
class ActivityListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView):
|
||||||
"""
|
"""
|
||||||
Displays all Activities, and classify if they are on-going or upcoming ones.
|
Displays all Activities, and classify if they are on-going or upcoming ones.
|
||||||
"""
|
"""
|
||||||
model = Activity
|
model = Activity
|
||||||
tables = [
|
table_class = ActivityTable
|
||||||
lambda data: ActivityTable(data, prefix="all-"),
|
ordering = ('-date_start',)
|
||||||
lambda data: ActivityTable(data, prefix="upcoming-"),
|
|
||||||
]
|
|
||||||
extra_context = {"title": _("Activities")}
|
extra_context = {"title": _("Activities")}
|
||||||
|
|
||||||
def get_queryset(self, **kwargs):
|
def get_queryset(self):
|
||||||
return super().get_queryset(**kwargs).distinct()
|
return super().get_queryset().distinct()
|
||||||
|
|
||||||
def get_tables_data(self):
|
|
||||||
# first table = all activities, second table = upcoming
|
|
||||||
return [
|
|
||||||
self.get_queryset().order_by("-date_start"),
|
|
||||||
Activity.objects.filter(date_end__gt=timezone.now())
|
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, Activity, "view"))
|
|
||||||
.distinct()
|
|
||||||
.order_by("date_start")
|
|
||||||
]
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
context = super().get_context_data(**kwargs)
|
context = super().get_context_data(**kwargs)
|
||||||
|
|
||||||
tables = context["tables"]
|
upcoming_activities = Activity.objects.filter(date_end__gt=timezone.now())
|
||||||
for name, table in zip(["table", "upcoming"], tables):
|
context['upcoming'] = ActivityTable(
|
||||||
context[name] = table
|
data=upcoming_activities.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view")),
|
||||||
|
prefix='upcoming-',
|
||||||
|
)
|
||||||
|
|
||||||
started_activities = self.get_queryset().filter(open=True, valid=True).distinct().all()
|
started_activities = Activity.objects\
|
||||||
|
.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view"))\
|
||||||
|
.filter(open=True, valid=True).all()
|
||||||
context["started_activities"] = started_activities
|
context["started_activities"] = started_activities
|
||||||
|
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
class ActivityDetailView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, DetailView):
|
class ActivityDetailView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
|
||||||
"""
|
"""
|
||||||
Shows details about one activity. Add guest to context
|
Shows details about one activity. Add guest to context
|
||||||
"""
|
"""
|
||||||
@ -104,40 +94,15 @@ class ActivityDetailView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMix
|
|||||||
context_object_name = "activity"
|
context_object_name = "activity"
|
||||||
extra_context = {"title": _("Activity detail")}
|
extra_context = {"title": _("Activity detail")}
|
||||||
|
|
||||||
tables = [
|
|
||||||
lambda data: GuestTable(data, prefix="guests-"),
|
|
||||||
lambda data: OpenerTable(data, prefix="opener-"),
|
|
||||||
]
|
|
||||||
|
|
||||||
def get_tables_data(self):
|
|
||||||
return [
|
|
||||||
Guest.objects.filter(activity=self.object)
|
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, Guest, "view")),
|
|
||||||
self.object.opener.filter(activity=self.object)
|
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, Opener, "view")),
|
|
||||||
]
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
def get_context_data(self, **kwargs):
|
||||||
context = super().get_context_data()
|
context = super().get_context_data()
|
||||||
|
|
||||||
tables = context["tables"]
|
table = GuestTable(data=Guest.objects.filter(activity=self.object)
|
||||||
for name, table in zip(["guests", "opener"], tables):
|
.filter(PermissionBackend.filter_queryset(self.request.user, Guest, "view")))
|
||||||
context[name] = table
|
context["guests"] = table
|
||||||
|
|
||||||
context["activity_started"] = timezone.now() > timezone.localtime(self.object.date_start)
|
context["activity_started"] = timezone.now() > timezone.localtime(self.object.date_start)
|
||||||
|
|
||||||
context["widget"] = {
|
|
||||||
"name": "opener",
|
|
||||||
"resetable": True,
|
|
||||||
"attrs": {
|
|
||||||
"class": "autocomplete form-control",
|
|
||||||
"id": "opener",
|
|
||||||
"api_url": "/api/note/alias/?note__polymorphic_ctype__model=noteuser",
|
|
||||||
"name_field": "name",
|
|
||||||
"placeholder": ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return context
|
return context
|
||||||
|
|
||||||
|
|
||||||
@ -179,41 +144,36 @@ class ActivityInviteView(ProtectQuerysetMixin, ProtectedCreateView):
|
|||||||
|
|
||||||
def get_form(self, form_class=None):
|
def get_form(self, form_class=None):
|
||||||
form = super().get_form(form_class)
|
form = super().get_form(form_class)
|
||||||
form.activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request, Activity, "view"))\
|
form.activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view"))\
|
||||||
.filter(pk=self.kwargs["pk"]).first()
|
.get(pk=self.kwargs["pk"])
|
||||||
form.fields["inviter"].initial = self.request.user.note
|
form.fields["inviter"].initial = self.request.user.note
|
||||||
return form
|
return form
|
||||||
|
|
||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def form_valid(self, form):
|
def form_valid(self, form):
|
||||||
form.instance.activity = Activity.objects\
|
form.instance.activity = Activity.objects\
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, Activity, "view")).get(pk=self.kwargs["pk"])
|
.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view")).get(pk=self.kwargs["pk"])
|
||||||
return super().form_valid(form)
|
return super().form_valid(form)
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
def get_success_url(self, **kwargs):
|
||||||
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.kwargs["pk"]})
|
return reverse_lazy('activity:activity_detail', kwargs={"pk": self.kwargs["pk"]})
|
||||||
|
|
||||||
|
|
||||||
class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView):
|
class ActivityEntryView(LoginRequiredMixin, TemplateView):
|
||||||
"""
|
"""
|
||||||
Manages entry to an activity
|
Manages entry to an activity
|
||||||
"""
|
"""
|
||||||
template_name = "activity/activity_entry.html"
|
template_name = "activity/activity_entry.html"
|
||||||
|
|
||||||
table_class = EntryTable
|
|
||||||
|
|
||||||
def dispatch(self, request, *args, **kwargs):
|
def dispatch(self, request, *args, **kwargs):
|
||||||
"""
|
"""
|
||||||
Don't display the entry interface if the user has no right to see it (no right to add an entry for itself),
|
Don't display the entry interface if the user has no right to see it (no right to add an entry for itself),
|
||||||
it is closed or doesn't manage entries.
|
it is closed or doesn't manage entries.
|
||||||
"""
|
"""
|
||||||
if not self.request.user.is_authenticated:
|
|
||||||
return self.handle_no_permission()
|
|
||||||
|
|
||||||
activity = Activity.objects.get(pk=self.kwargs["pk"])
|
activity = Activity.objects.get(pk=self.kwargs["pk"])
|
||||||
|
|
||||||
sample_entry = Entry(activity=activity, note=self.request.user.note)
|
sample_entry = Entry(activity=activity, note=self.request.user.note)
|
||||||
if not PermissionBackend.check_perm(self.request, "activity.add_entry", sample_entry):
|
if not PermissionBackend.check_perm(self.request.user, "activity.add_entry", sample_entry):
|
||||||
raise PermissionDenied(_("You are not allowed to display the entry interface for this activity."))
|
raise PermissionDenied(_("You are not allowed to display the entry interface for this activity."))
|
||||||
|
|
||||||
if not activity.activity_type.manage_entries:
|
if not activity.activity_type.manage_entries:
|
||||||
@ -231,25 +191,22 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView):
|
|||||||
guest_qs = Guest.objects\
|
guest_qs = Guest.objects\
|
||||||
.annotate(balance=F("inviter__balance"), note_name=F("inviter__user__username"))\
|
.annotate(balance=F("inviter__balance"), note_name=F("inviter__user__username"))\
|
||||||
.filter(activity=activity)\
|
.filter(activity=activity)\
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, Guest, "view"))\
|
.filter(PermissionBackend.filter_queryset(self.request.user, Guest, "view"))\
|
||||||
.order_by('last_name', 'first_name')
|
.order_by('last_name', 'first_name').distinct()
|
||||||
|
|
||||||
if "search" in self.request.GET and self.request.GET["search"]:
|
if "search" in self.request.GET and self.request.GET["search"]:
|
||||||
pattern = self.request.GET["search"]
|
pattern = self.request.GET["search"]
|
||||||
|
if pattern[0] != "^":
|
||||||
# Check if this is a valid regex. If not, we won't check regex
|
pattern = "^" + pattern
|
||||||
valid_regex = is_regex(pattern)
|
|
||||||
suffix = "__iregex" if valid_regex else "__istartswith"
|
|
||||||
pattern = "^" + pattern if valid_regex and pattern[0] != "^" else pattern
|
|
||||||
guest_qs = guest_qs.filter(
|
guest_qs = guest_qs.filter(
|
||||||
Q(**{f"first_name{suffix}": pattern})
|
Q(first_name__iregex=pattern)
|
||||||
| Q(**{f"last_name{suffix}": pattern})
|
| Q(last_name__iregex=pattern)
|
||||||
| Q(**{f"inviter__alias__name{suffix}": pattern})
|
| Q(inviter__alias__name__iregex=pattern)
|
||||||
| Q(**{f"inviter__alias__normalized_name{suffix}": Alias.normalize(pattern)})
|
| Q(inviter__alias__normalized_name__iregex=Alias.normalize(pattern))
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
guest_qs = guest_qs.none()
|
guest_qs = guest_qs.none()
|
||||||
return guest_qs.distinct()
|
return guest_qs
|
||||||
|
|
||||||
def get_invited_note(self, activity):
|
def get_invited_note(self, activity):
|
||||||
"""
|
"""
|
||||||
@ -265,26 +222,23 @@ 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 valid members
|
# Keep only 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()).exclude(note__inactivity_reason='forced')
|
note__noteuser__user__memberships__date_end__gte=timezone.now(),
|
||||||
|
)
|
||||||
|
|
||||||
# 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.user, Alias, "view"))
|
||||||
|
|
||||||
if "search" in self.request.GET and self.request.GET["search"]:
|
if "search" in self.request.GET and self.request.GET["search"]:
|
||||||
pattern = self.request.GET["search"]
|
pattern = self.request.GET["search"]
|
||||||
|
|
||||||
# Check if this is a valid regex. If not, we won't check regex
|
|
||||||
valid_regex = is_regex(pattern)
|
|
||||||
suffix = "__iregex" if valid_regex else "__icontains"
|
|
||||||
note_qs = note_qs.filter(
|
note_qs = note_qs.filter(
|
||||||
Q(**{f"note__noteuser__user__first_name{suffix}": pattern})
|
Q(note__noteuser__user__first_name__iregex=pattern)
|
||||||
| Q(**{f"note__noteuser__user__last_name{suffix}": pattern})
|
| Q(note__noteuser__user__last_name__iregex=pattern)
|
||||||
| Q(**{f"name{suffix}": pattern})
|
| Q(name__iregex=pattern)
|
||||||
| Q(**{f"normalized_name{suffix}": Alias.normalize(pattern)})
|
| Q(normalized_name__iregex=Alias.normalize(pattern))
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
note_qs = note_qs.none()
|
note_qs = note_qs.none()
|
||||||
@ -296,9 +250,15 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView):
|
|||||||
if settings.DATABASES[note_qs.db]["ENGINE"] == 'django.db.backends.postgresql' else note_qs.distinct()[:20]
|
if settings.DATABASES[note_qs.db]["ENGINE"] == 'django.db.backends.postgresql' else note_qs.distinct()[:20]
|
||||||
return note_qs
|
return note_qs
|
||||||
|
|
||||||
def get_table_data(self):
|
def get_context_data(self, **kwargs):
|
||||||
activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request, Activity, "view"))\
|
"""
|
||||||
|
Query the list of Guest and Note to the activity and add information to makes entry with JS.
|
||||||
|
"""
|
||||||
|
context = super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request.user, Activity, "view"))\
|
||||||
.distinct().get(pk=self.kwargs["pk"])
|
.distinct().get(pk=self.kwargs["pk"])
|
||||||
|
context["activity"] = activity
|
||||||
|
|
||||||
matched = []
|
matched = []
|
||||||
|
|
||||||
@ -311,17 +271,8 @@ class ActivityEntryView(LoginRequiredMixin, SingleTableMixin, TemplateView):
|
|||||||
note.activity = activity
|
note.activity = activity
|
||||||
matched.append(note)
|
matched.append(note)
|
||||||
|
|
||||||
return matched
|
table = EntryTable(data=matched)
|
||||||
|
context["table"] = table
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
"""
|
|
||||||
Query the list of Guest and Note to the activity and add information to makes entry with JS.
|
|
||||||
"""
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
activity = Activity.objects.filter(PermissionBackend.filter_queryset(self.request, Activity, "view"))\
|
|
||||||
.distinct().get(pk=self.kwargs["pk"])
|
|
||||||
context["activity"] = activity
|
|
||||||
|
|
||||||
context["entries"] = Entry.objects.filter(activity=activity)
|
context["entries"] = Entry.objects.filter(activity=activity)
|
||||||
|
|
||||||
@ -329,10 +280,10 @@ 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, activity_type__manage_entries=True).filter(
|
activities_open = Activity.objects.filter(open=True).filter(
|
||||||
PermissionBackend.filter_queryset(self.request, Activity, "view")).distinct().all()
|
PermissionBackend.filter_queryset(self.request.user, 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.user,
|
||||||
"activity.add_entry",
|
"activity.add_entry",
|
||||||
Entry(activity=a, note=self.request.user.note,))]
|
Entry(activity=a, note=self.request.user.note,))]
|
||||||
|
|
||||||
@ -363,8 +314,8 @@ X-WR-CALNAME:Kfet Calendar
|
|||||||
NAME:Kfet Calendar
|
NAME:Kfet Calendar
|
||||||
CALSCALE:GREGORIAN
|
CALSCALE:GREGORIAN
|
||||||
BEGIN:VTIMEZONE
|
BEGIN:VTIMEZONE
|
||||||
TZID:Europe/Paris
|
TZID:Europe/Berlin
|
||||||
X-LIC-LOCATION:Europe/Paris
|
X-LIC-LOCATION:Europe/Berlin
|
||||||
BEGIN:DAYLIGHT
|
BEGIN:DAYLIGHT
|
||||||
TZOFFSETFROM:+0100
|
TZOFFSETFROM:+0100
|
||||||
TZOFFSETTO:+0200
|
TZOFFSETTO:+0200
|
||||||
@ -386,10 +337,10 @@ END:VTIMEZONE
|
|||||||
DTSTAMP:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)}Z
|
DTSTAMP:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)}Z
|
||||||
UID:{md5((activity.name + "$" + str(activity.id) + str(activity.date_start)).encode("UTF-8")).hexdigest()}
|
UID:{md5((activity.name + "$" + str(activity.id) + str(activity.date_start)).encode("UTF-8")).hexdigest()}
|
||||||
SUMMARY;CHARSET=UTF-8:{self.multilines(activity.name, 75, 22)}
|
SUMMARY;CHARSET=UTF-8:{self.multilines(activity.name, 75, 22)}
|
||||||
DTSTART:{"{:%Y%m%dT%H%M%S}Z".format(activity.date_start)}
|
DTSTART;TZID=Europe/Berlin:{"{:%Y%m%dT%H%M%S}".format(activity.date_start)}
|
||||||
DTEND:{"{:%Y%m%dT%H%M%S}Z".format(activity.date_end)}
|
DTEND;TZID=Europe/Berlin:{"{:%Y%m%dT%H%M%S}".format(activity.date_end)}
|
||||||
LOCATION:{self.multilines(activity.location, 75, 9) if activity.location else "Kfet"}
|
LOCATION:{self.multilines(activity.location, 75, 9) if activity.location else "Kfet"}
|
||||||
DESCRIPTION;CHARSET=UTF-8:""" + self.multilines(activity.description.replace("\n", "\\n"), 75, 26) + f"""
|
DESCRIPTION;CHARSET=UTF-8:""" + self.multilines(activity.description.replace("\n", "\\n"), 75, 26) + """
|
||||||
-- {activity.organizer.name}
|
-- {activity.organizer.name}
|
||||||
END:VEVENT
|
END:VEVENT
|
||||||
"""
|
"""
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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,42 +0,0 @@
|
|||||||
import re
|
|
||||||
from functools import lru_cache
|
|
||||||
|
|
||||||
from rest_framework.filters import SearchFilter
|
|
||||||
|
|
||||||
|
|
||||||
class RegexSafeSearchFilter(SearchFilter):
|
|
||||||
@lru_cache
|
|
||||||
def validate_regex(self, search_term) -> bool:
|
|
||||||
try:
|
|
||||||
re.compile(search_term)
|
|
||||||
return True
|
|
||||||
except re.error:
|
|
||||||
return False
|
|
||||||
|
|
||||||
def get_search_fields(self, view, request):
|
|
||||||
"""
|
|
||||||
Ensure that given regex are valid.
|
|
||||||
If not, we consider that the user is trying to search by substring.
|
|
||||||
"""
|
|
||||||
search_fields = super().get_search_fields(view, request)
|
|
||||||
search_terms = self.get_search_terms(request)
|
|
||||||
|
|
||||||
for search_term in search_terms:
|
|
||||||
if not self.validate_regex(search_term):
|
|
||||||
# Invalid regex. We assume we don't query by regex but by substring.
|
|
||||||
search_fields = [f.replace('$', '') for f in search_fields]
|
|
||||||
break
|
|
||||||
|
|
||||||
return search_fields
|
|
||||||
|
|
||||||
def get_search_terms(self, request):
|
|
||||||
"""
|
|
||||||
Ensure that search field is a valid regex query. If not, we remove extra characters.
|
|
||||||
"""
|
|
||||||
terms = super().get_search_terms(request)
|
|
||||||
if not all(self.validate_regex(term) for term in terms):
|
|
||||||
# Invalid regex. If a ^ is prefixed to the search term, we remove it.
|
|
||||||
terms = [term[1:] if term[0] == '^' else term for term in terms]
|
|
||||||
# Same for dollars.
|
|
||||||
terms = [term[:-1] if term[-1] == '$' else term for term in terms]
|
|
||||||
return terms
|
|
@ -1,5 +0,0 @@
|
|||||||
from rest_framework.pagination import PageNumberPagination
|
|
||||||
|
|
||||||
|
|
||||||
class CustomPagination(PageNumberPagination):
|
|
||||||
page_size_query_param = 'page_size'
|
|
@ -1,20 +1,13 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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 django.utils import timezone
|
from rest_framework.serializers import ModelSerializer
|
||||||
from rest_framework import serializers
|
|
||||||
from member.api.serializers import ProfileSerializer, MembershipSerializer
|
|
||||||
from member.models import Membership
|
|
||||||
from note.api.serializers import NoteSerializer
|
|
||||||
from note.models import Alias
|
|
||||||
from note_kfet.middlewares import get_current_request
|
|
||||||
from permission.backends import PermissionBackend
|
|
||||||
|
|
||||||
|
|
||||||
class UserSerializer(serializers.ModelSerializer):
|
class UserSerializer(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.
|
||||||
@ -29,7 +22,7 @@ class UserSerializer(serializers.ModelSerializer):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class ContentTypeSerializer(serializers.ModelSerializer):
|
class ContentTypeSerializer(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.
|
||||||
@ -38,54 +31,3 @@ class ContentTypeSerializer(serializers.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 = serializers.SerializerMethodField()
|
|
||||||
|
|
||||||
note = serializers.SerializerMethodField()
|
|
||||||
|
|
||||||
memberships = serializers.SerializerMethodField()
|
|
||||||
|
|
||||||
def get_normalized_name(self, obj):
|
|
||||||
return Alias.normalize(obj.username)
|
|
||||||
|
|
||||||
def get_profile(self, obj):
|
|
||||||
# Display the profile of the user only if we have rights to see it.
|
|
||||||
return ProfileSerializer().to_representation(obj.profile) \
|
|
||||||
if PermissionBackend.check_perm(get_current_request(), 'member.view_profile', obj.profile) else None
|
|
||||||
|
|
||||||
def get_note(self, obj):
|
|
||||||
# Display the note of the user only if we have rights to see it.
|
|
||||||
return NoteSerializer().to_representation(obj.note) \
|
|
||||||
if PermissionBackend.check_perm(get_current_request(), 'note.view_note', obj.note) else None
|
|
||||||
|
|
||||||
def get_memberships(self, obj):
|
|
||||||
# Display only memberships that we are allowed to see.
|
|
||||||
return serializers.ListSerializer(child=MembershipSerializer()).to_representation(
|
|
||||||
obj.memberships.filter(date_start__lte=timezone.now(), date_end__gte=timezone.now())
|
|
||||||
.filter(PermissionBackend.filter_queryset(get_current_request(), Membership, 'view')))
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = User
|
|
||||||
fields = (
|
|
||||||
'id',
|
|
||||||
'username',
|
|
||||||
'normalized_name',
|
|
||||||
'first_name',
|
|
||||||
'last_name',
|
|
||||||
'email',
|
|
||||||
'is_superuser',
|
|
||||||
'is_active',
|
|
||||||
'is_staff',
|
|
||||||
'profile',
|
|
||||||
'note',
|
|
||||||
'memberships',
|
|
||||||
)
|
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
|
|
||||||
@ -12,12 +11,11 @@ from django.contrib.contenttypes.models import ContentType
|
|||||||
from django.db.models.fields.files import ImageFieldFile
|
from django.db.models.fields.files import ImageFieldFile
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
from django_filters.rest_framework import DjangoFilterBackend
|
from django_filters.rest_framework import DjangoFilterBackend
|
||||||
from phonenumbers import PhoneNumber
|
|
||||||
from rest_framework.filters import OrderingFilter
|
|
||||||
from api.filters import RegexSafeSearchFilter
|
|
||||||
from member.models import Membership, Club
|
from member.models import Membership, Club
|
||||||
from note.models import NoteClub, NoteUser, Alias, Note
|
from note.models import NoteClub, NoteUser, Alias, Note
|
||||||
from permission.models import PermissionMask, Permission, Role
|
from permission.models import PermissionMask, Permission, Role
|
||||||
|
from phonenumbers import PhoneNumber
|
||||||
|
from rest_framework.filters import SearchFilter, OrderingFilter
|
||||||
|
|
||||||
from .viewsets import ContentTypeViewSet, UserViewSet
|
from .viewsets import ContentTypeViewSet, UserViewSet
|
||||||
|
|
||||||
@ -88,7 +86,7 @@ class TestAPI(TestCase):
|
|||||||
resp = self.client.get(url + f"?ordering=-{field}")
|
resp = self.client.get(url + f"?ordering=-{field}")
|
||||||
self.assertEqual(resp.status_code, 200)
|
self.assertEqual(resp.status_code, 200)
|
||||||
|
|
||||||
if RegexSafeSearchFilter in backends:
|
if SearchFilter in backends:
|
||||||
# Basic search
|
# Basic search
|
||||||
for field in viewset.search_fields:
|
for field in viewset.search_fields:
|
||||||
obj = self.fix_note_object(obj, field)
|
obj = self.fix_note_object(obj, field)
|
||||||
@ -154,8 +152,6 @@ 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,12 +1,10 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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 include
|
from django.conf.urls import url, include
|
||||||
from django.urls import re_path
|
|
||||||
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.
|
||||||
@ -15,48 +13,39 @@ router = routers.DefaultRouter()
|
|||||||
router.register('models', ContentTypeViewSet)
|
router.register('models', ContentTypeViewSet)
|
||||||
router.register('user', UserViewSet)
|
router.register('user', UserViewSet)
|
||||||
|
|
||||||
if "activity" in settings.INSTALLED_APPS:
|
|
||||||
from activity.api.urls import register_activity_urls
|
|
||||||
register_activity_urls(router, 'activity')
|
|
||||||
|
|
||||||
if "food" in settings.INSTALLED_APPS:
|
|
||||||
from food.api.urls import register_food_urls
|
|
||||||
register_food_urls(router, 'food')
|
|
||||||
|
|
||||||
if "logs" in settings.INSTALLED_APPS:
|
|
||||||
from logs.api.urls import register_logs_urls
|
|
||||||
register_logs_urls(router, 'logs')
|
|
||||||
|
|
||||||
if "member" in settings.INSTALLED_APPS:
|
if "member" in settings.INSTALLED_APPS:
|
||||||
from member.api.urls import register_members_urls
|
from member.api.urls import register_members_urls
|
||||||
register_members_urls(router, 'members')
|
register_members_urls(router, 'members')
|
||||||
|
|
||||||
|
if "member" in settings.INSTALLED_APPS:
|
||||||
|
from activity.api.urls import register_activity_urls
|
||||||
|
register_activity_urls(router, 'activity')
|
||||||
|
|
||||||
if "note" in settings.INSTALLED_APPS:
|
if "note" in settings.INSTALLED_APPS:
|
||||||
from note.api.urls import register_note_urls
|
from note.api.urls import register_note_urls
|
||||||
register_note_urls(router, 'note')
|
register_note_urls(router, 'note')
|
||||||
|
|
||||||
if "permission" in settings.INSTALLED_APPS:
|
|
||||||
from permission.api.urls import register_permission_urls
|
|
||||||
register_permission_urls(router, 'permission')
|
|
||||||
|
|
||||||
if "treasury" in settings.INSTALLED_APPS:
|
if "treasury" in settings.INSTALLED_APPS:
|
||||||
from treasury.api.urls import register_treasury_urls
|
from treasury.api.urls import register_treasury_urls
|
||||||
register_treasury_urls(router, 'treasury')
|
register_treasury_urls(router, 'treasury')
|
||||||
|
|
||||||
|
if "permission" in settings.INSTALLED_APPS:
|
||||||
|
from permission.api.urls import register_permission_urls
|
||||||
|
register_permission_urls(router, 'permission')
|
||||||
|
|
||||||
|
if "logs" in settings.INSTALLED_APPS:
|
||||||
|
from logs.api.urls import register_logs_urls
|
||||||
|
register_logs_urls(router, 'logs')
|
||||||
|
|
||||||
if "wei" in settings.INSTALLED_APPS:
|
if "wei" in settings.INSTALLED_APPS:
|
||||||
from wei.api.urls import register_wei_urls
|
from wei.api.urls import register_wei_urls
|
||||||
register_wei_urls(router, 'wei')
|
register_wei_urls(router, 'wei')
|
||||||
|
|
||||||
if "wrapped" in settings.INSTALLED_APPS:
|
|
||||||
from wrapped.api.urls import register_wrapped_urls
|
|
||||||
register_wrapped_urls(router, 'wrapped')
|
|
||||||
|
|
||||||
app_name = 'api'
|
app_name = 'api'
|
||||||
|
|
||||||
# Wire up our API using automatic URL routing.
|
# Wire up our API using automatic URL routing.
|
||||||
# Additionally, we include login URLs for the browsable API.
|
# Additionally, we include login URLs for the browsable API.
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
re_path('^', include(router.urls)),
|
url('^', include(router.urls)),
|
||||||
re_path('^me/', UserInformationView.as_view()),
|
url('^api-auth/', include('rest_framework.urls', namespace='rest_framework')),
|
||||||
re_path('^api-auth/', include('rest_framework.urls', namespace='rest_framework')),
|
|
||||||
]
|
]
|
||||||
|
@ -1,20 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 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,29 +1,20 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
import re
|
|
||||||
|
|
||||||
from django.contrib.contenttypes.models import ContentType
|
from django.contrib.contenttypes.models import ContentType
|
||||||
from django_filters.rest_framework import DjangoFilterBackend
|
from django_filters.rest_framework import DjangoFilterBackend
|
||||||
from django.db.models import Q
|
from django.db.models import Q
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.auth.models import User
|
from django.contrib.auth.models import User
|
||||||
|
from rest_framework.filters import SearchFilter
|
||||||
from rest_framework.viewsets import ReadOnlyModelViewSet, ModelViewSet
|
from rest_framework.viewsets import ReadOnlyModelViewSet, ModelViewSet
|
||||||
from permission.backends import PermissionBackend
|
from permission.backends import PermissionBackend
|
||||||
|
from note_kfet.middlewares import get_current_session
|
||||||
from note.models import Alias
|
from note.models import Alias
|
||||||
|
|
||||||
from .filters import RegexSafeSearchFilter
|
|
||||||
from .serializers import UserSerializer, ContentTypeSerializer
|
from .serializers import UserSerializer, ContentTypeSerializer
|
||||||
|
|
||||||
|
|
||||||
def is_regex(pattern):
|
|
||||||
try:
|
|
||||||
re.compile(pattern)
|
|
||||||
return True
|
|
||||||
except (re.error, TypeError):
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class ReadProtectedModelViewSet(ModelViewSet):
|
class ReadProtectedModelViewSet(ModelViewSet):
|
||||||
"""
|
"""
|
||||||
Protect a ModelViewSet by filtering the objects that the user cannot see.
|
Protect a ModelViewSet by filtering the objects that the user cannot see.
|
||||||
@ -34,7 +25,9 @@ class ReadProtectedModelViewSet(ModelViewSet):
|
|||||||
self.model = ContentType.objects.get_for_model(self.serializer_class.Meta.model).model_class()
|
self.model = ContentType.objects.get_for_model(self.serializer_class.Meta.model).model_class()
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
return self.queryset.filter(PermissionBackend.filter_queryset(self.request, self.model, "view")).distinct()
|
user = self.request.user
|
||||||
|
get_current_session().setdefault("permission_mask", 42)
|
||||||
|
return self.queryset.filter(PermissionBackend.filter_queryset(user, self.model, "view")).distinct()
|
||||||
|
|
||||||
|
|
||||||
class ReadOnlyProtectedModelViewSet(ReadOnlyModelViewSet):
|
class ReadOnlyProtectedModelViewSet(ReadOnlyModelViewSet):
|
||||||
@ -47,7 +40,9 @@ class ReadOnlyProtectedModelViewSet(ReadOnlyModelViewSet):
|
|||||||
self.model = ContentType.objects.get_for_model(self.serializer_class.Meta.model).model_class()
|
self.model = ContentType.objects.get_for_model(self.serializer_class.Meta.model).model_class()
|
||||||
|
|
||||||
def get_queryset(self):
|
def get_queryset(self):
|
||||||
return self.queryset.filter(PermissionBackend.filter_queryset(self.request, self.model, "view")).distinct()
|
user = self.request.user
|
||||||
|
get_current_session().setdefault("permission_mask", 42)
|
||||||
|
return self.queryset.filter(PermissionBackend.filter_queryset(user, self.model, "view")).distinct()
|
||||||
|
|
||||||
|
|
||||||
class UserViewSet(ReadProtectedModelViewSet):
|
class UserViewSet(ReadProtectedModelViewSet):
|
||||||
@ -70,38 +65,34 @@ class UserViewSet(ReadProtectedModelViewSet):
|
|||||||
|
|
||||||
if "search" in self.request.GET:
|
if "search" in self.request.GET:
|
||||||
pattern = self.request.GET["search"]
|
pattern = self.request.GET["search"]
|
||||||
# Check if this is a valid regex. If not, we won't check regex
|
|
||||||
valid_regex = is_regex(pattern)
|
|
||||||
suffix = "__iregex" if valid_regex else "__istartswith"
|
|
||||||
prefix = "^" if valid_regex else ""
|
|
||||||
|
|
||||||
# Filter with different rules
|
# Filter with different rules
|
||||||
# We use union-all to keep each filter rule sorted in result
|
# We use union-all to keep each filter rule sorted in result
|
||||||
queryset = queryset.filter(
|
queryset = queryset.filter(
|
||||||
# Match without normalization
|
# Match without normalization
|
||||||
Q(**{f"note__alias__name{suffix}": prefix + pattern})
|
note__alias__name__iregex="^" + pattern
|
||||||
).union(
|
).union(
|
||||||
queryset.filter(
|
queryset.filter(
|
||||||
# Match with normalization
|
# Match with normalization
|
||||||
Q(**{f"note__alias__normalized_name{suffix}": prefix + Alias.normalize(pattern)})
|
Q(note__alias__normalized_name__iregex="^" + Alias.normalize(pattern))
|
||||||
& ~Q(**{f"note__alias__name{suffix}": prefix + pattern})
|
& ~Q(note__alias__name__iregex="^" + pattern)
|
||||||
),
|
),
|
||||||
all=True,
|
all=True,
|
||||||
).union(
|
).union(
|
||||||
queryset.filter(
|
queryset.filter(
|
||||||
# Match on lower pattern
|
# Match on lower pattern
|
||||||
Q(**{f"note__alias__normalized_name{suffix}": prefix + pattern.lower()})
|
Q(note__alias__normalized_name__iregex="^" + pattern.lower())
|
||||||
& ~Q(**{f"note__alias__normalized_name{suffix}": prefix + Alias.normalize(pattern)})
|
& ~Q(note__alias__normalized_name__iregex="^" + Alias.normalize(pattern))
|
||||||
& ~Q(**{f"note__alias__name{suffix}": prefix + pattern})
|
& ~Q(note__alias__name__iregex="^" + pattern)
|
||||||
),
|
),
|
||||||
all=True,
|
all=True,
|
||||||
).union(
|
).union(
|
||||||
queryset.filter(
|
queryset.filter(
|
||||||
# Match on firstname or lastname
|
# Match on firstname or lastname
|
||||||
(Q(**{f"last_name{suffix}": prefix + pattern}) | Q(**{f"first_name{suffix}": prefix + pattern}))
|
(Q(last_name__iregex="^" + pattern) | Q(first_name__iregex="^" + pattern))
|
||||||
& ~Q(**{f"note__alias__normalized_name{suffix}": prefix + pattern.lower()})
|
& ~Q(note__alias__normalized_name__iregex="^" + pattern.lower())
|
||||||
& ~Q(**{f"note__alias__normalized_name{suffix}": prefix + Alias.normalize(pattern)})
|
& ~Q(note__alias__normalized_name__iregex="^" + Alias.normalize(pattern))
|
||||||
& ~Q(**{f"note__alias__name{suffix}": prefix + pattern})
|
& ~Q(note__alias__name__iregex="^" + pattern)
|
||||||
),
|
),
|
||||||
all=True,
|
all=True,
|
||||||
)
|
)
|
||||||
@ -121,6 +112,6 @@ class ContentTypeViewSet(ReadOnlyModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = ContentType.objects.order_by('id')
|
queryset = ContentType.objects.order_by('id')
|
||||||
serializer_class = ContentTypeSerializer
|
serializer_class = ContentTypeSerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['id', 'app_label', 'model', ]
|
filterset_fields = ['id', 'app_label', 'model', ]
|
||||||
search_fields = ['$app_label', '$model', ]
|
search_fields = ['$app_label', '$model', ]
|
||||||
|
@ -1,37 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from django.contrib import admin
|
|
||||||
from django.db import transaction
|
|
||||||
from note_kfet.admin import admin_site
|
|
||||||
|
|
||||||
from .models import Allergen, BasicFood, QRCode, TransformedFood
|
|
||||||
|
|
||||||
|
|
||||||
@admin.register(QRCode, site=admin_site)
|
|
||||||
class QRCodeAdmin(admin.ModelAdmin):
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
@admin.register(BasicFood, site=admin_site)
|
|
||||||
class BasicFoodAdmin(admin.ModelAdmin):
|
|
||||||
@transaction.atomic
|
|
||||||
def save_related(self, *args, **kwargs):
|
|
||||||
ans = super().save_related(*args, **kwargs)
|
|
||||||
args[1].instance.update()
|
|
||||||
return ans
|
|
||||||
|
|
||||||
|
|
||||||
@admin.register(TransformedFood, site=admin_site)
|
|
||||||
class TransformedFoodAdmin(admin.ModelAdmin):
|
|
||||||
exclude = ["allergens", "expiry_date"]
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def save_related(self, request, form, *args, **kwargs):
|
|
||||||
super().save_related(request, form, *args, **kwargs)
|
|
||||||
form.instance.update()
|
|
||||||
|
|
||||||
|
|
||||||
@admin.register(Allergen, site=admin_site)
|
|
||||||
class AllergenAdmin(admin.ModelAdmin):
|
|
||||||
pass
|
|
@ -1,50 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from rest_framework import serializers
|
|
||||||
|
|
||||||
from ..models import Allergen, BasicFood, QRCode, TransformedFood
|
|
||||||
|
|
||||||
|
|
||||||
class AllergenSerializer(serializers.ModelSerializer):
|
|
||||||
"""
|
|
||||||
REST API Serializer for Allergen.
|
|
||||||
The djangorestframework plugin will analyse the model `Allergen` and parse all fields in the API.
|
|
||||||
"""
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = Allergen
|
|
||||||
fields = '__all__'
|
|
||||||
|
|
||||||
|
|
||||||
class BasicFoodSerializer(serializers.ModelSerializer):
|
|
||||||
"""
|
|
||||||
REST API Serializer for BasicFood.
|
|
||||||
The djangorestframework plugin will analyse the model `BasicFood` and parse all fields in the API.
|
|
||||||
"""
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = BasicFood
|
|
||||||
fields = '__all__'
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeSerializer(serializers.ModelSerializer):
|
|
||||||
"""
|
|
||||||
REST API Serializer for QRCode.
|
|
||||||
The djangorestframework plugin will analyse the model `QRCode` and parse all fields in the API.
|
|
||||||
"""
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = QRCode
|
|
||||||
fields = '__all__'
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodSerializer(serializers.ModelSerializer):
|
|
||||||
"""
|
|
||||||
REST API Serializer for TransformedFood.
|
|
||||||
The djangorestframework plugin will analyse the model `TransformedFood` and parse all fields in the API.
|
|
||||||
"""
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = TransformedFood
|
|
||||||
fields = '__all__'
|
|
@ -1,14 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from .views import AllergenViewSet, BasicFoodViewSet, QRCodeViewSet, TransformedFoodViewSet
|
|
||||||
|
|
||||||
|
|
||||||
def register_food_urls(router, path):
|
|
||||||
"""
|
|
||||||
Configure router for Food REST API.
|
|
||||||
"""
|
|
||||||
router.register(path + '/allergen', AllergenViewSet)
|
|
||||||
router.register(path + '/basic_food', BasicFoodViewSet)
|
|
||||||
router.register(path + '/qrcode', QRCodeViewSet)
|
|
||||||
router.register(path + '/transformed_food', TransformedFoodViewSet)
|
|
@ -1,61 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from api.viewsets import ReadProtectedModelViewSet
|
|
||||||
from django_filters.rest_framework import DjangoFilterBackend
|
|
||||||
from rest_framework.filters import SearchFilter
|
|
||||||
|
|
||||||
from .serializers import AllergenSerializer, BasicFoodSerializer, QRCodeSerializer, TransformedFoodSerializer
|
|
||||||
from ..models import Allergen, BasicFood, QRCode, TransformedFood
|
|
||||||
|
|
||||||
|
|
||||||
class AllergenViewSet(ReadProtectedModelViewSet):
|
|
||||||
"""
|
|
||||||
REST API View set.
|
|
||||||
The djangorestframework plugin will get all `Allergen` objects, serialize it to JSON with the given serializer,
|
|
||||||
then render it on /api/food/allergen/
|
|
||||||
"""
|
|
||||||
queryset = Allergen.objects.order_by('id')
|
|
||||||
serializer_class = AllergenSerializer
|
|
||||||
filter_backends = [DjangoFilterBackend, SearchFilter]
|
|
||||||
filterset_fields = ['name', ]
|
|
||||||
search_fields = ['$name', ]
|
|
||||||
|
|
||||||
|
|
||||||
class BasicFoodViewSet(ReadProtectedModelViewSet):
|
|
||||||
"""
|
|
||||||
REST API View set.
|
|
||||||
The djangorestframework plugin will get all `BasicFood` objects, serialize it to JSON with the given serializer,
|
|
||||||
then render it on /api/food/basic_food/
|
|
||||||
"""
|
|
||||||
queryset = BasicFood.objects.order_by('id')
|
|
||||||
serializer_class = BasicFoodSerializer
|
|
||||||
filter_backends = [DjangoFilterBackend, SearchFilter]
|
|
||||||
filterset_fields = ['name', ]
|
|
||||||
search_fields = ['$name', ]
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeViewSet(ReadProtectedModelViewSet):
|
|
||||||
"""
|
|
||||||
REST API View set.
|
|
||||||
The djangorestframework plugin will get all `QRCode` objects, serialize it to JSON with the given serializer,
|
|
||||||
then render it on /api/food/qrcode/
|
|
||||||
"""
|
|
||||||
queryset = QRCode.objects.order_by('id')
|
|
||||||
serializer_class = QRCodeSerializer
|
|
||||||
filter_backends = [DjangoFilterBackend, SearchFilter]
|
|
||||||
filterset_fields = ['qr_code_number', ]
|
|
||||||
search_fields = ['$qr_code_number', ]
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodViewSet(ReadProtectedModelViewSet):
|
|
||||||
"""
|
|
||||||
REST API View set.
|
|
||||||
The djangorestframework plugin will get all `TransformedFood` objects, serialize it to JSON with the given serializer,
|
|
||||||
then render it on /api/food/transformed_food/
|
|
||||||
"""
|
|
||||||
queryset = TransformedFood.objects.order_by('id')
|
|
||||||
serializer_class = TransformedFoodSerializer
|
|
||||||
filter_backends = [DjangoFilterBackend, SearchFilter]
|
|
||||||
filterset_fields = ['name', ]
|
|
||||||
search_fields = ['$name', ]
|
|
@ -1,11 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
|
||||||
from django.apps import AppConfig
|
|
||||||
|
|
||||||
|
|
||||||
class FoodkfetConfig(AppConfig):
|
|
||||||
name = 'food'
|
|
||||||
verbose_name = _('food')
|
|
@ -1,114 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from random import shuffle
|
|
||||||
|
|
||||||
from django import forms
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
|
||||||
from django.utils import timezone
|
|
||||||
from member.models import Club
|
|
||||||
from bootstrap_datepicker_plus.widgets import DateTimePickerInput
|
|
||||||
from note_kfet.inputs import Autocomplete
|
|
||||||
from note_kfet.middlewares import get_current_request
|
|
||||||
from permission.backends import PermissionBackend
|
|
||||||
|
|
||||||
from .models import BasicFood, QRCode, TransformedFood
|
|
||||||
|
|
||||||
|
|
||||||
class AddIngredientForms(forms.ModelForm):
|
|
||||||
"""
|
|
||||||
Form for add an ingredient
|
|
||||||
"""
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.fields['ingredient'].queryset = self.fields['ingredient'].queryset.filter(
|
|
||||||
polymorphic_ctype__model='transformedfood',
|
|
||||||
is_ready=False,
|
|
||||||
is_active=True,
|
|
||||||
was_eaten=False,
|
|
||||||
)
|
|
||||||
# Caution, the logic is inverted here, we flip the logic on saving in AddIngredientView
|
|
||||||
self.fields['is_active'].initial = True
|
|
||||||
self.fields['is_active'].label = _("Fully used")
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = TransformedFood
|
|
||||||
fields = ('ingredient', 'is_active')
|
|
||||||
|
|
||||||
|
|
||||||
class BasicFoodForms(forms.ModelForm):
|
|
||||||
"""
|
|
||||||
Form for add non-transformed food
|
|
||||||
"""
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.fields['name'].widget.attrs.update({"autofocus": "autofocus"})
|
|
||||||
self.fields['name'].required = True
|
|
||||||
self.fields['owner'].required = True
|
|
||||||
|
|
||||||
# Some example
|
|
||||||
self.fields['name'].widget.attrs.update({"placeholder": _("Pasta METRO 5kg")})
|
|
||||||
clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all())
|
|
||||||
shuffle(clubs)
|
|
||||||
self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = BasicFood
|
|
||||||
fields = ('name', 'owner', 'date_type', 'expiry_date', 'is_active', 'was_eaten', 'allergens',)
|
|
||||||
widgets = {
|
|
||||||
"owner": Autocomplete(
|
|
||||||
model=Club,
|
|
||||||
attrs={"api_url": "/api/members/club/"},
|
|
||||||
),
|
|
||||||
'expiry_date': DateTimePickerInput(),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeForms(forms.ModelForm):
|
|
||||||
"""
|
|
||||||
Form for create QRCode
|
|
||||||
"""
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.fields['food_container'].queryset = self.fields['food_container'].queryset.filter(
|
|
||||||
is_active=True,
|
|
||||||
was_eaten=False,
|
|
||||||
polymorphic_ctype__model='transformedfood',
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = QRCode
|
|
||||||
fields = ('food_container',)
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodForms(forms.ModelForm):
|
|
||||||
"""
|
|
||||||
Form for add transformed food
|
|
||||||
"""
|
|
||||||
def __init__(self, *args, **kwargs):
|
|
||||||
super().__init__(*args, **kwargs)
|
|
||||||
self.fields['name'].widget.attrs.update({"autofocus": "autofocus"})
|
|
||||||
self.fields['name'].required = True
|
|
||||||
self.fields['owner'].required = True
|
|
||||||
self.fields['creation_date'].required = True
|
|
||||||
self.fields['creation_date'].initial = timezone.now
|
|
||||||
self.fields['is_active'].initial = True
|
|
||||||
self.fields['is_ready'].initial = False
|
|
||||||
self.fields['was_eaten'].initial = False
|
|
||||||
|
|
||||||
# Some example
|
|
||||||
self.fields['name'].widget.attrs.update({"placeholder": _("Lasagna")})
|
|
||||||
clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all())
|
|
||||||
shuffle(clubs)
|
|
||||||
self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = TransformedFood
|
|
||||||
fields = ('name', 'creation_date', 'owner', 'is_active', 'is_ready', 'was_eaten', 'shelf_life')
|
|
||||||
widgets = {
|
|
||||||
"owner": Autocomplete(
|
|
||||||
model=Club,
|
|
||||||
attrs={"api_url": "/api/members/club/"},
|
|
||||||
),
|
|
||||||
'creation_date': DateTimePickerInput(),
|
|
||||||
}
|
|
@ -1,84 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-07-05 08:57
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
import django.db.models.deletion
|
|
||||||
import django.utils.timezone
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
initial = True
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('contenttypes', '0002_remove_content_type_name'),
|
|
||||||
('member', '0011_profile_vss_charter_read'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='Allergen',
|
|
||||||
fields=[
|
|
||||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
||||||
('name', models.CharField(max_length=255, verbose_name='name')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'Allergen',
|
|
||||||
'verbose_name_plural': 'Allergens',
|
|
||||||
},
|
|
||||||
),
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='Food',
|
|
||||||
fields=[
|
|
||||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
||||||
('name', models.CharField(max_length=255, verbose_name='name')),
|
|
||||||
('expiry_date', models.DateTimeField(verbose_name='expiry date')),
|
|
||||||
('was_eaten', models.BooleanField(default=False, verbose_name='was eaten')),
|
|
||||||
('is_ready', models.BooleanField(default=False, verbose_name='is ready')),
|
|
||||||
('allergens', models.ManyToManyField(blank=True, to='food.Allergen', verbose_name='allergen')),
|
|
||||||
('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='+', to='member.Club', verbose_name='owner')),
|
|
||||||
('polymorphic_ctype', models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_food.food_set+', to='contenttypes.ContentType')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'foods',
|
|
||||||
},
|
|
||||||
),
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='BasicFood',
|
|
||||||
fields=[
|
|
||||||
('food_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='food.Food')),
|
|
||||||
('date_type', models.CharField(choices=[('DLC', 'DLC'), ('DDM', 'DDM')], max_length=255)),
|
|
||||||
('arrival_date', models.DateTimeField(default=django.utils.timezone.now, verbose_name='arrival date')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'Basic food',
|
|
||||||
'verbose_name_plural': 'Basic foods',
|
|
||||||
},
|
|
||||||
bases=('food.food',),
|
|
||||||
),
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='QRCode',
|
|
||||||
fields=[
|
|
||||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
|
||||||
('qr_code_number', models.PositiveIntegerField(unique=True, verbose_name='QR-code number')),
|
|
||||||
('food_container', models.OneToOneField(on_delete=django.db.models.deletion.PROTECT, related_name='QR_code', to='food.Food', verbose_name='food container')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'QR-code',
|
|
||||||
'verbose_name_plural': 'QR-codes',
|
|
||||||
},
|
|
||||||
),
|
|
||||||
migrations.CreateModel(
|
|
||||||
name='TransformedFood',
|
|
||||||
fields=[
|
|
||||||
('food_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='food.Food')),
|
|
||||||
('creation_date', models.DateTimeField(verbose_name='creation date')),
|
|
||||||
('is_active', models.BooleanField(default=True, verbose_name='is active')),
|
|
||||||
('ingredient', models.ManyToManyField(blank=True, related_name='transformed_ingredient_inv', to='food.Food', verbose_name='transformed ingredient')),
|
|
||||||
],
|
|
||||||
options={
|
|
||||||
'verbose_name': 'Transformed food',
|
|
||||||
'verbose_name_plural': 'Transformed foods',
|
|
||||||
},
|
|
||||||
bases=('food.food',),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,19 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-07-06 20:37
|
|
||||||
|
|
||||||
import datetime
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('food', '0001_initial'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name='transformedfood',
|
|
||||||
name='shelf_life',
|
|
||||||
field=models.DurationField(default=datetime.timedelta(days=3), verbose_name='shelf life'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,62 +0,0 @@
|
|||||||
from django.db import migrations
|
|
||||||
|
|
||||||
def create_14_mandatory_allergens(apps, schema_editor):
|
|
||||||
"""
|
|
||||||
There are 14 mandatory allergens, they are pre-injected
|
|
||||||
"""
|
|
||||||
|
|
||||||
Allergen = apps.get_model("food", "allergen")
|
|
||||||
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Gluten",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Fruits à coques",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Crustacés",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Céléri",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Oeufs",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Moutarde",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Poissons",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Soja",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Lait",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Sulfites",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Sésame",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Lupin",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Arachides",
|
|
||||||
)
|
|
||||||
Allergen.objects.get_or_create(
|
|
||||||
name="Mollusques",
|
|
||||||
)
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
dependencies = [
|
|
||||||
('food', '0002_transformedfood_shelf_life'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RunPython(create_14_mandatory_allergens),
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
@ -1,28 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-08-13 21:58
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
import django.db.models.deletion
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('food', '0003_create_14_allergens_mandatory'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.RemoveField(
|
|
||||||
model_name='transformedfood',
|
|
||||||
name='is_active',
|
|
||||||
),
|
|
||||||
migrations.AddField(
|
|
||||||
model_name='food',
|
|
||||||
name='is_active',
|
|
||||||
field=models.BooleanField(default=True, verbose_name='is active'),
|
|
||||||
),
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='qrcode',
|
|
||||||
name='food_container',
|
|
||||||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='QR_code', to='food.Food', verbose_name='food container'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,20 +0,0 @@
|
|||||||
# Generated by Django 4.2.15 on 2024-08-28 08:00
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
import django.db.models.deletion
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('contenttypes', '0002_remove_content_type_name'),
|
|
||||||
('food', '0004_auto_20240813_2358'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='food',
|
|
||||||
name='polymorphic_ctype',
|
|
||||||
field=models.ForeignKey(editable=False, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='polymorphic_%(app_label)s.%(class)s_set+', to='contenttypes.contenttype'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,226 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
|
|
||||||
from django.db import models, transaction
|
|
||||||
from django.utils import timezone
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
|
||||||
from member.models import Club
|
|
||||||
from polymorphic.models import PolymorphicModel
|
|
||||||
|
|
||||||
|
|
||||||
class QRCode(models.Model):
|
|
||||||
"""
|
|
||||||
An QRCode model
|
|
||||||
"""
|
|
||||||
qr_code_number = models.PositiveIntegerField(
|
|
||||||
verbose_name=_("QR-code number"),
|
|
||||||
unique=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
food_container = models.ForeignKey(
|
|
||||||
'Food',
|
|
||||||
on_delete=models.CASCADE,
|
|
||||||
related_name='QR_code',
|
|
||||||
verbose_name=_('food container'),
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _("QR-code")
|
|
||||||
verbose_name_plural = _("QR-codes")
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return _("QR-code number {qr_code_number}").format(qr_code_number=self.qr_code_number)
|
|
||||||
|
|
||||||
|
|
||||||
class Allergen(models.Model):
|
|
||||||
"""
|
|
||||||
A list of allergen and alimentary restrictions
|
|
||||||
"""
|
|
||||||
name = models.CharField(
|
|
||||||
verbose_name=_('name'),
|
|
||||||
max_length=255,
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('Allergen')
|
|
||||||
verbose_name_plural = _('Allergens')
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
|
|
||||||
class Food(PolymorphicModel):
|
|
||||||
name = models.CharField(
|
|
||||||
verbose_name=_('name'),
|
|
||||||
max_length=255,
|
|
||||||
)
|
|
||||||
|
|
||||||
owner = models.ForeignKey(
|
|
||||||
Club,
|
|
||||||
on_delete=models.PROTECT,
|
|
||||||
related_name='+',
|
|
||||||
verbose_name=_('owner'),
|
|
||||||
)
|
|
||||||
|
|
||||||
allergens = models.ManyToManyField(
|
|
||||||
Allergen,
|
|
||||||
blank=True,
|
|
||||||
verbose_name=_('allergen'),
|
|
||||||
)
|
|
||||||
|
|
||||||
expiry_date = models.DateTimeField(
|
|
||||||
verbose_name=_('expiry date'),
|
|
||||||
null=False,
|
|
||||||
)
|
|
||||||
|
|
||||||
was_eaten = models.BooleanField(
|
|
||||||
default=False,
|
|
||||||
verbose_name=_('was eaten'),
|
|
||||||
)
|
|
||||||
|
|
||||||
# is_ready != is_active : is_ready signifie que la nourriture est prête à être manger,
|
|
||||||
# is_active signifie que la nourriture n'est pas encore archivé
|
|
||||||
# il sert dans les cas où il est plus intéressant que de l'open soit conservé (confiture par ex)
|
|
||||||
|
|
||||||
is_ready = models.BooleanField(
|
|
||||||
default=False,
|
|
||||||
verbose_name=_('is ready'),
|
|
||||||
)
|
|
||||||
|
|
||||||
is_active = models.BooleanField(
|
|
||||||
default=True,
|
|
||||||
verbose_name=_('is active'),
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
|
|
||||||
return super().save(force_insert, force_update, using, update_fields)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('food')
|
|
||||||
verbose_name = _('foods')
|
|
||||||
|
|
||||||
|
|
||||||
class BasicFood(Food):
|
|
||||||
"""
|
|
||||||
Food which has been directly buy on supermarket
|
|
||||||
"""
|
|
||||||
date_type = models.CharField(
|
|
||||||
max_length=255,
|
|
||||||
choices=(
|
|
||||||
("DLC", "DLC"),
|
|
||||||
("DDM", "DDM"),
|
|
||||||
)
|
|
||||||
)
|
|
||||||
|
|
||||||
arrival_date = models.DateTimeField(
|
|
||||||
verbose_name=_('arrival date'),
|
|
||||||
default=timezone.now,
|
|
||||||
)
|
|
||||||
|
|
||||||
# label = models.ImageField(
|
|
||||||
# verbose_name=_('food label'),
|
|
||||||
# max_length=255,
|
|
||||||
# blank=False,
|
|
||||||
# null=False,
|
|
||||||
# upload_to='label/',
|
|
||||||
# )
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update_allergens(self):
|
|
||||||
# update parents
|
|
||||||
for parent in self.transformed_ingredient_inv.iterator():
|
|
||||||
parent.update_allergens()
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update_expiry_date(self):
|
|
||||||
# update parents
|
|
||||||
for parent in self.transformed_ingredient_inv.iterator():
|
|
||||||
parent.update_expiry_date()
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update(self):
|
|
||||||
self.update_allergens()
|
|
||||||
self.update_expiry_date()
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('Basic food')
|
|
||||||
verbose_name_plural = _('Basic foods')
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFood(Food):
|
|
||||||
"""
|
|
||||||
Transformed food are a mix between basic food and meal
|
|
||||||
"""
|
|
||||||
creation_date = models.DateTimeField(
|
|
||||||
verbose_name=_('creation date'),
|
|
||||||
)
|
|
||||||
|
|
||||||
ingredient = models.ManyToManyField(
|
|
||||||
Food,
|
|
||||||
blank=True,
|
|
||||||
symmetrical=False,
|
|
||||||
related_name='transformed_ingredient_inv',
|
|
||||||
verbose_name=_('transformed ingredient'),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Without microbiological analyzes, the storage time is 3 days
|
|
||||||
shelf_life = models.DurationField(
|
|
||||||
verbose_name=_("shelf life"),
|
|
||||||
default=timedelta(days=3),
|
|
||||||
)
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def archive(self):
|
|
||||||
# When a meal are archived, if it was eaten, update ingredient fully used for this meal
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update_allergens(self):
|
|
||||||
# When allergens are changed, simply update the parents' allergens
|
|
||||||
old_allergens = list(self.allergens.all())
|
|
||||||
self.allergens.clear()
|
|
||||||
for ingredient in self.ingredient.iterator():
|
|
||||||
self.allergens.set(self.allergens.union(ingredient.allergens.all()))
|
|
||||||
|
|
||||||
if old_allergens == list(self.allergens.all()):
|
|
||||||
return
|
|
||||||
super().save()
|
|
||||||
|
|
||||||
# update parents
|
|
||||||
for parent in self.transformed_ingredient_inv.iterator():
|
|
||||||
parent.update_allergens()
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update_expiry_date(self):
|
|
||||||
# When expiry_date is changed, simply update the parents' expiry_date
|
|
||||||
old_expiry_date = self.expiry_date
|
|
||||||
self.expiry_date = self.creation_date + self.shelf_life
|
|
||||||
for ingredient in self.ingredient.iterator():
|
|
||||||
self.expiry_date = min(self.expiry_date, ingredient.expiry_date)
|
|
||||||
|
|
||||||
if old_expiry_date == self.expiry_date:
|
|
||||||
return
|
|
||||||
super().save()
|
|
||||||
|
|
||||||
# update parents
|
|
||||||
for parent in self.transformed_ingredient_inv.iterator():
|
|
||||||
parent.update_expiry_date()
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def update(self):
|
|
||||||
self.update_allergens()
|
|
||||||
self.update_expiry_date()
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def save(self, *args, **kwargs):
|
|
||||||
super().save(*args, **kwargs)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('Transformed food')
|
|
||||||
verbose_name_plural = _('Transformed foods')
|
|
@ -1,19 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
import django_tables2 as tables
|
|
||||||
from django_tables2 import A
|
|
||||||
|
|
||||||
from .models import TransformedFood
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodTable(tables.Table):
|
|
||||||
name = tables.LinkColumn(
|
|
||||||
'food:food_view',
|
|
||||||
args=[A('pk'), ],
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
model = TransformedFood
|
|
||||||
template_name = 'django_tables2/bootstrap4.html'
|
|
||||||
fields = ('name', "owner", "allergens", "expiry_date")
|
|
@ -1,20 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body" id="form">
|
|
||||||
<form method="post">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form|crispy }}
|
|
||||||
<button class="btn btn-primary" type="submit">{% trans "Submit"%}</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,37 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }} {{ food.name }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body">
|
|
||||||
<ul>
|
|
||||||
<li><p>{% trans 'Owner' %} : {{ food.owner }}</p></li>
|
|
||||||
<li><p>{% trans 'Arrival date' %} : {{ food.arrival_date }}</p></li>
|
|
||||||
<li><p>{% trans 'Expiry date' %} : {{ food.expiry_date }} ({{ food.date_type }})</p></li>
|
|
||||||
<li>{% trans 'Allergens' %} :</li>
|
|
||||||
<ul>
|
|
||||||
{% for allergen in food.allergens.iterator %}
|
|
||||||
<li>{{ allergen.name }}</li>
|
|
||||||
{% endfor %}
|
|
||||||
</ul>
|
|
||||||
<p>
|
|
||||||
<li><p>{% trans 'Active' %} : {{ food.is_active }}<p></li>
|
|
||||||
<li><p>{% trans 'Eaten' %} : {{ food.was_eaten }}<p></li>
|
|
||||||
</ul>
|
|
||||||
{% if can_update %}
|
|
||||||
<a class="btn btn-sm btn-warning" href="{% url "food:basic_update" pk=food.pk %}">{% trans 'Update' %}</a>
|
|
||||||
{% endif %}
|
|
||||||
{% if can_add_ingredient %}
|
|
||||||
<a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=food.pk %}">
|
|
||||||
{% trans 'Add to a meal' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,20 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body" id="form">
|
|
||||||
<form method="post">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form | crispy }}
|
|
||||||
<button class="btn btn-primary" type="submit">{% trans "Submit"%}</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,55 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load render_table from django_tables2 %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body" id="form">
|
|
||||||
<a class="btn btn-sm btn-success" href="{% url "food:qrcode_basic_create" slug=slug %}">
|
|
||||||
{% trans 'New basic food' %}
|
|
||||||
</a>
|
|
||||||
<form method="post">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form|crispy }}
|
|
||||||
<button class="btn btn-primary" type="submit">{% trans "Submit" %}</button>
|
|
||||||
</form>
|
|
||||||
<div class="card-body" id="profile_infos">
|
|
||||||
<h4>{% trans "Copy constructor" %}</h4>
|
|
||||||
<table class="table">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th class="orderable">
|
|
||||||
{% trans "Name" %}
|
|
||||||
</th>
|
|
||||||
<th class="orderable">
|
|
||||||
{% trans "Owner" %}
|
|
||||||
</th>
|
|
||||||
<th class="orderable">
|
|
||||||
{% trans "Arrival date" %}
|
|
||||||
</th>
|
|
||||||
<th class="orderable">
|
|
||||||
{% trans "Expiry date" %}
|
|
||||||
</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody>
|
|
||||||
{% for basic in last_basic %}
|
|
||||||
<tr>
|
|
||||||
<td><a href="{% url "food:qrcode_basic_create" slug=slug %}?copy={{ basic.pk }}">{{ basic.name }}</a></td>
|
|
||||||
<td>{{ basic.owner }}</td>
|
|
||||||
<td>{{ basic.arrival_date }}</td>
|
|
||||||
<td>{{ basic.expiry_date }}</td>
|
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,39 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }} {% trans 'number' %} {{ qrcode.qr_code_number }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body">
|
|
||||||
<ul>
|
|
||||||
<li><p>{% trans 'Name' %} : {{ qrcode.food_container.name }}</p></li>
|
|
||||||
<li><p>{% trans 'Owner' %} : {{ qrcode.food_container.owner }}</p></li>
|
|
||||||
<li><p>{% trans 'Expiry date' %} : {{ qrcode.food_container.expiry_date }}</p></li>
|
|
||||||
</ul>
|
|
||||||
{% if qrcode.food_container.polymorphic_ctype.model == 'basicfood' and can_update_basic %}
|
|
||||||
<a class="btn btn-sm btn-warning" href="{% url "food:basic_update" pk=qrcode.food_container.pk %}" data-turbolinks="false">
|
|
||||||
{% trans 'Update' %}
|
|
||||||
</a>
|
|
||||||
{% elif can_update_transformed %}
|
|
||||||
<a class="btn btn-sm btn-warning" href="{% url "food:transformed_update" pk=qrcode.food_container.pk %}">
|
|
||||||
{% trans 'Update' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
{% if can_view_detail %}
|
|
||||||
<a class="btn btn-sm btn-primary" href="{% url "food:food_view" pk=qrcode.food_container.pk %}">
|
|
||||||
{% trans 'View details' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
{% if can_add_ingredient %}
|
|
||||||
<a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=qrcode.food_container.pk %}">
|
|
||||||
{% trans 'Add to a meal' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,51 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }} {{ food.name }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body">
|
|
||||||
<ul>
|
|
||||||
<li><p>{% trans 'Owner' %} : {{ food.owner }}</p></li>
|
|
||||||
{% if can_see_ready %}
|
|
||||||
<li><p>{% trans 'Ready' %} : {{ food.is_ready }}</p></li>
|
|
||||||
{% endif %}
|
|
||||||
<li><p>{% trans 'Creation date' %} : {{ food.creation_date }}</p></li>
|
|
||||||
<li><p>{% trans 'Expiry date' %} : {{ food.expiry_date }}</p></li>
|
|
||||||
<li>{% trans 'Allergens' %} :</li>
|
|
||||||
<ul>
|
|
||||||
{% for allergen in food.allergens.iterator %}
|
|
||||||
<li>{{ allergen.name }}</li>
|
|
||||||
{% endfor %}
|
|
||||||
</ul>
|
|
||||||
<p>
|
|
||||||
<li>{% trans 'Ingredients' %} :</li>
|
|
||||||
<ul>
|
|
||||||
{% for ingredient in food.ingredient.iterator %}
|
|
||||||
<li><a href="{% url "food:food_view" pk=ingredient.pk %}">{{ ingredient.name }}</a></li>
|
|
||||||
{% endfor %}
|
|
||||||
</ul>
|
|
||||||
<p>
|
|
||||||
<li><p>{% trans 'Shelf life' %} : {{ food.shelf_life }}</p></li>
|
|
||||||
<li><p>{% trans 'Ready' %} : {{ food.is_ready }}</p></li>
|
|
||||||
<li><p>{% trans 'Active' %} : {{ food.is_active }}</p></li>
|
|
||||||
<li><p>{% trans 'Eaten' %} : {{ food.was_eaten }}</p></li>
|
|
||||||
</ul>
|
|
||||||
{% if can_update %}
|
|
||||||
<a class="btn btn-sm btn-warning" href="{% url "food:transformed_update" pk=food.pk %}">
|
|
||||||
{% trans 'Update' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
{% if can_add_ingredient %}
|
|
||||||
<a class="btn btn-sm btn-success" href="{% url "food:add_ingredient" pk=food.pk %}">
|
|
||||||
{% trans 'Add to a meal' %}
|
|
||||||
</a>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,20 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load i18n crispy_forms_tags %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-white mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{{ title }}
|
|
||||||
</h3>
|
|
||||||
<div class="card-body" id="form">
|
|
||||||
<form method="post">
|
|
||||||
{% csrf_token %}
|
|
||||||
{{ form|crispy }}
|
|
||||||
<button class="btn btn-primary" type="submit">{% trans "Submit"%}</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,60 +0,0 @@
|
|||||||
{% extends "base.html" %}
|
|
||||||
{% comment %}
|
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
{% endcomment %}
|
|
||||||
{% load render_table from django_tables2 %}
|
|
||||||
{% load i18n %}
|
|
||||||
|
|
||||||
{% block content %}
|
|
||||||
<div class="card bg-light mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{% trans "Meal served" %}
|
|
||||||
</h3>
|
|
||||||
{% if can_create_meal %}
|
|
||||||
<div class="card-footer">
|
|
||||||
<a class="btn btn-sm btn-success" href="{% url 'food:transformed_create' %}" data-turbolinks="false">
|
|
||||||
{% trans 'New meal' %}
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
{% if served.data %}
|
|
||||||
{% render_table served %}
|
|
||||||
{% else %}
|
|
||||||
<div class="card-body">
|
|
||||||
<div class="alert alert-warning">
|
|
||||||
{% trans "There is no meal served." %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="card bg-light mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{% trans "Open" %}
|
|
||||||
</h3>
|
|
||||||
{% if open.data %}
|
|
||||||
{% render_table open %}
|
|
||||||
{% else %}
|
|
||||||
<div class="card-body">
|
|
||||||
<div class="alert alert-warning">
|
|
||||||
{% trans "There is no free meal." %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="card bg-light mb-3">
|
|
||||||
<h3 class="card-header text-center">
|
|
||||||
{% trans "All meals" %}
|
|
||||||
</h3>
|
|
||||||
{% if table.data %}
|
|
||||||
{% render_table table %}
|
|
||||||
{% else %}
|
|
||||||
<div class="card-body">
|
|
||||||
<div class="alert alert-warning">
|
|
||||||
{% trans "There is no meal." %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
{% endblock %}
|
|
@ -1,3 +0,0 @@
|
|||||||
# from django.test import TestCase
|
|
||||||
|
|
||||||
# Create your tests here.
|
|
@ -1,21 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from django.urls import path
|
|
||||||
|
|
||||||
from . import views
|
|
||||||
|
|
||||||
app_name = 'food'
|
|
||||||
|
|
||||||
urlpatterns = [
|
|
||||||
path('', views.TransformedListView.as_view(), name='food_list'),
|
|
||||||
path('<int:slug>', views.QRCodeView.as_view(), name='qrcode_view'),
|
|
||||||
path('detail/<int:pk>', views.FoodView.as_view(), name='food_view'),
|
|
||||||
|
|
||||||
path('<int:slug>/create_qrcode', views.QRCodeCreateView.as_view(), name='qrcode_create'),
|
|
||||||
path('<int:slug>/create_qrcode/basic', views.QRCodeBasicFoodCreateView.as_view(), name='qrcode_basic_create'),
|
|
||||||
path('create/transformed', views.TransformedFoodCreateView.as_view(), name='transformed_create'),
|
|
||||||
path('update/basic/<int:pk>', views.BasicFoodUpdateView.as_view(), name='basic_update'),
|
|
||||||
path('update/transformed/<int:pk>', views.TransformedFoodUpdateView.as_view(), name='transformed_update'),
|
|
||||||
path('add/<int:pk>', views.AddIngredientView.as_view(), name='add_ingredient'),
|
|
||||||
]
|
|
@ -1,421 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
|
|
||||||
from django.db import transaction
|
|
||||||
from django.contrib.auth.mixins import LoginRequiredMixin
|
|
||||||
from django.http import HttpResponseRedirect
|
|
||||||
from django_tables2.views import MultiTableMixin
|
|
||||||
from django.urls import reverse
|
|
||||||
from django.utils.translation import gettext_lazy as _
|
|
||||||
from django.utils import timezone
|
|
||||||
from django.views.generic import DetailView, UpdateView
|
|
||||||
from django.views.generic.list import ListView
|
|
||||||
from django.forms import HiddenInput
|
|
||||||
from permission.backends import PermissionBackend
|
|
||||||
from permission.views import ProtectQuerysetMixin, ProtectedCreateView
|
|
||||||
|
|
||||||
from .forms import AddIngredientForms, BasicFoodForms, QRCodeForms, TransformedFoodForms
|
|
||||||
from .models import BasicFood, Food, QRCode, TransformedFood
|
|
||||||
from .tables import TransformedFoodTable
|
|
||||||
|
|
||||||
|
|
||||||
class AddIngredientView(ProtectQuerysetMixin, UpdateView):
|
|
||||||
"""
|
|
||||||
A view to add an ingredient
|
|
||||||
"""
|
|
||||||
model = Food
|
|
||||||
template_name = 'food/add_ingredient_form.html'
|
|
||||||
extra_context = {"title": _("Add the ingredient")}
|
|
||||||
form_class = AddIngredientForms
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
context["pk"] = self.kwargs["pk"]
|
|
||||||
return context
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
food = Food.objects.get(pk=self.kwargs['pk'])
|
|
||||||
add_ingredient_form = AddIngredientForms(data=self.request.POST)
|
|
||||||
if food.is_ready:
|
|
||||||
form.add_error(None, _("The product is already prepared"))
|
|
||||||
return self.form_invalid(form)
|
|
||||||
if not add_ingredient_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
# We flip logic ""fully used = not is_active""
|
|
||||||
food.is_active = not food.is_active
|
|
||||||
# Save the aliment and the allergens associed
|
|
||||||
for transformed_pk in self.request.POST.getlist('ingredient'):
|
|
||||||
transformed = TransformedFood.objects.get(pk=transformed_pk)
|
|
||||||
if not transformed.is_ready:
|
|
||||||
transformed.ingredient.add(food)
|
|
||||||
transformed.update()
|
|
||||||
food.save()
|
|
||||||
|
|
||||||
return HttpResponseRedirect(self.get_success_url())
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
return reverse('food:food_list')
|
|
||||||
|
|
||||||
|
|
||||||
class BasicFoodUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
|
|
||||||
"""
|
|
||||||
A view to update a basic food
|
|
||||||
"""
|
|
||||||
model = BasicFood
|
|
||||||
form_class = BasicFoodForms
|
|
||||||
template_name = 'food/basicfood_form.html'
|
|
||||||
extra_context = {"title": _("Update an aliment")}
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
basic_food_form = BasicFoodForms(data=self.request.POST)
|
|
||||||
if not basic_food_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
ans = super().form_valid(form)
|
|
||||||
form.instance.update()
|
|
||||||
return ans
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
self.object.refresh_from_db()
|
|
||||||
return reverse('food:food_view', kwargs={"pk": self.object.pk})
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class FoodView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
|
|
||||||
"""
|
|
||||||
A view to see a food
|
|
||||||
"""
|
|
||||||
model = Food
|
|
||||||
extra_context = {"title": _("Details of:")}
|
|
||||||
context_object_name = "food"
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
context["can_update"] = PermissionBackend.check_perm(self.request, "food.change_food")
|
|
||||||
context["can_add_ingredient"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood")
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeBasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
|
||||||
#####################################################################
|
|
||||||
# TO DO
|
|
||||||
# - this feature is very pratical for meat or fish, nevertheless we can implement this later
|
|
||||||
# - fix picture save
|
|
||||||
# - implement solution crop and convert image (reuse or recode ImageForm from members apps)
|
|
||||||
#####################################################################
|
|
||||||
"""
|
|
||||||
A view to add a basic food with a qrcode
|
|
||||||
"""
|
|
||||||
model = BasicFood
|
|
||||||
form_class = BasicFoodForms
|
|
||||||
template_name = 'food/basicfood_form.html'
|
|
||||||
extra_context = {"title": _("Add a new basic food with QRCode")}
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
basic_food_form = BasicFoodForms(data=self.request.POST)
|
|
||||||
if not basic_food_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
# Save the aliment and the allergens associed
|
|
||||||
basic_food = form.save(commit=False)
|
|
||||||
# We assume the date of labeling and the same as the date of arrival
|
|
||||||
basic_food.arrival_date = timezone.now()
|
|
||||||
basic_food.is_ready = False
|
|
||||||
basic_food.is_active = True
|
|
||||||
basic_food.was_eaten = False
|
|
||||||
basic_food._force_save = True
|
|
||||||
basic_food.save()
|
|
||||||
basic_food.refresh_from_db()
|
|
||||||
|
|
||||||
qrcode = QRCode()
|
|
||||||
qrcode.qr_code_number = self.kwargs['slug']
|
|
||||||
qrcode.food_container = basic_food
|
|
||||||
qrcode.save()
|
|
||||||
|
|
||||||
return super().form_valid(form)
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
self.object.refresh_from_db()
|
|
||||||
return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']})
|
|
||||||
|
|
||||||
def get_sample_object(self):
|
|
||||||
|
|
||||||
# We choose a club which may work or BDE else
|
|
||||||
owner_id = 1
|
|
||||||
for membership in self.request.user.memberships.all():
|
|
||||||
club_id = membership.club.id
|
|
||||||
food = BasicFood(name="", expiry_date=timezone.now(), owner_id=club_id)
|
|
||||||
if PermissionBackend.check_perm(self.request, "food.add_basicfood", food):
|
|
||||||
owner_id = club_id
|
|
||||||
|
|
||||||
return BasicFood(
|
|
||||||
name="",
|
|
||||||
expiry_date=timezone.now(),
|
|
||||||
owner_id=owner_id,
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
# Some field are hidden on create
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
form = context['form']
|
|
||||||
form.fields['is_active'].widget = HiddenInput()
|
|
||||||
form.fields['was_eaten'].widget = HiddenInput()
|
|
||||||
|
|
||||||
copy = self.request.GET.get('copy', None)
|
|
||||||
if copy is not None:
|
|
||||||
basic = BasicFood.objects.get(pk=copy)
|
|
||||||
for field in ['date_type', 'expiry_date', 'name', 'owner']:
|
|
||||||
form.fields[field].initial = getattr(basic, field)
|
|
||||||
for field in ['allergens']:
|
|
||||||
form.fields[field].initial = getattr(basic, field).all()
|
|
||||||
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
|
||||||
"""
|
|
||||||
A view to add a new qrcode
|
|
||||||
"""
|
|
||||||
model = QRCode
|
|
||||||
template_name = 'food/create_qrcode_form.html'
|
|
||||||
form_class = QRCodeForms
|
|
||||||
extra_context = {"title": _("Add a new QRCode")}
|
|
||||||
|
|
||||||
def get(self, *args, **kwargs):
|
|
||||||
qrcode = kwargs["slug"]
|
|
||||||
if self.model.objects.filter(qr_code_number=qrcode).count() > 0:
|
|
||||||
return HttpResponseRedirect(reverse("food:qrcode_view", kwargs=kwargs))
|
|
||||||
else:
|
|
||||||
return super().get(*args, **kwargs)
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
context["slug"] = self.kwargs["slug"]
|
|
||||||
|
|
||||||
context["last_basic"] = BasicFood.objects.order_by('-pk').all()[:10]
|
|
||||||
|
|
||||||
return context
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
qrcode_food_form = QRCodeForms(data=self.request.POST)
|
|
||||||
if not qrcode_food_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
# Save the qrcode
|
|
||||||
qrcode = form.save(commit=False)
|
|
||||||
qrcode.qr_code_number = self.kwargs["slug"]
|
|
||||||
qrcode._force_save = True
|
|
||||||
qrcode.save()
|
|
||||||
qrcode.refresh_from_db()
|
|
||||||
|
|
||||||
qrcode.food_container.save()
|
|
||||||
|
|
||||||
return super().form_valid(form)
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
self.object.refresh_from_db()
|
|
||||||
return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']})
|
|
||||||
|
|
||||||
def get_sample_object(self):
|
|
||||||
return QRCode(
|
|
||||||
qr_code_number=self.kwargs["slug"],
|
|
||||||
food_container_id=1
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
class QRCodeView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView):
|
|
||||||
"""
|
|
||||||
A view to see a qrcode
|
|
||||||
"""
|
|
||||||
model = QRCode
|
|
||||||
extra_context = {"title": _("QRCode")}
|
|
||||||
context_object_name = "qrcode"
|
|
||||||
slug_field = "qr_code_number"
|
|
||||||
|
|
||||||
def get(self, *args, **kwargs):
|
|
||||||
qrcode = kwargs["slug"]
|
|
||||||
if self.model.objects.filter(qr_code_number=qrcode).count() > 0:
|
|
||||||
return super().get(*args, **kwargs)
|
|
||||||
else:
|
|
||||||
return HttpResponseRedirect(reverse("food:qrcode_create", kwargs=kwargs))
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
qr_code_number = self.kwargs['slug']
|
|
||||||
qrcode = self.model.objects.get(qr_code_number=qr_code_number)
|
|
||||||
|
|
||||||
model = qrcode.food_container.polymorphic_ctype.model
|
|
||||||
|
|
||||||
if model == "basicfood":
|
|
||||||
context["can_update_basic"] = PermissionBackend.check_perm(self.request, "food.change_basicfood")
|
|
||||||
context["can_view_detail"] = PermissionBackend.check_perm(self.request, "food.view_basicfood")
|
|
||||||
if model == "transformedfood":
|
|
||||||
context["can_update_transformed"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood")
|
|
||||||
context["can_view_detail"] = PermissionBackend.check_perm(self.request, "food.view_transformedfood")
|
|
||||||
context["can_add_ingredient"] = PermissionBackend.check_perm(self.request, "food.change_transformedfood")
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
|
||||||
"""
|
|
||||||
A view to add a tranformed food
|
|
||||||
"""
|
|
||||||
model = TransformedFood
|
|
||||||
template_name = 'food/transformedfood_form.html'
|
|
||||||
form_class = TransformedFoodForms
|
|
||||||
extra_context = {"title": _("Add a new meal")}
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
transformed_food_form = TransformedFoodForms(data=self.request.POST)
|
|
||||||
if not transformed_food_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
# Save the aliment and allergens associated
|
|
||||||
transformed_food = form.save(commit=False)
|
|
||||||
transformed_food.expiry_date = transformed_food.creation_date
|
|
||||||
transformed_food.is_active = True
|
|
||||||
transformed_food.is_ready = False
|
|
||||||
transformed_food.was_eaten = False
|
|
||||||
transformed_food._force_save = True
|
|
||||||
transformed_food.save()
|
|
||||||
transformed_food.refresh_from_db()
|
|
||||||
ans = super().form_valid(form)
|
|
||||||
transformed_food.update()
|
|
||||||
return ans
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
self.object.refresh_from_db()
|
|
||||||
return reverse('food:food_view', kwargs={"pk": self.object.pk})
|
|
||||||
|
|
||||||
def get_sample_object(self):
|
|
||||||
# We choose a club which may work or BDE else
|
|
||||||
owner_id = 1
|
|
||||||
for membership in self.request.user.memberships.all():
|
|
||||||
club_id = membership.club.id
|
|
||||||
food = TransformedFood(name="",
|
|
||||||
creation_date=timezone.now(),
|
|
||||||
expiry_date=timezone.now(),
|
|
||||||
owner_id=club_id)
|
|
||||||
if PermissionBackend.check_perm(self.request, "food.add_transformedfood", food):
|
|
||||||
owner_id = club_id
|
|
||||||
break
|
|
||||||
|
|
||||||
return TransformedFood(
|
|
||||||
name="",
|
|
||||||
owner_id=owner_id,
|
|
||||||
creation_date=timezone.now(),
|
|
||||||
expiry_date=timezone.now(),
|
|
||||||
)
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
# Some field are hidden on create
|
|
||||||
form = context['form']
|
|
||||||
form.fields['is_active'].widget = HiddenInput()
|
|
||||||
form.fields['is_ready'].widget = HiddenInput()
|
|
||||||
form.fields['was_eaten'].widget = HiddenInput()
|
|
||||||
form.fields['shelf_life'].widget = HiddenInput()
|
|
||||||
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedFoodUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView):
|
|
||||||
"""
|
|
||||||
A view to update transformed product
|
|
||||||
"""
|
|
||||||
model = TransformedFood
|
|
||||||
template_name = 'food/transformedfood_form.html'
|
|
||||||
form_class = TransformedFoodForms
|
|
||||||
extra_context = {'title': _('Update a meal')}
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def form_valid(self, form):
|
|
||||||
form.instance.creater = self.request.user
|
|
||||||
transformedfood_form = TransformedFoodForms(data=self.request.POST)
|
|
||||||
if not transformedfood_form.is_valid():
|
|
||||||
return self.form_invalid(form)
|
|
||||||
|
|
||||||
ans = super().form_valid(form)
|
|
||||||
form.instance.update()
|
|
||||||
return ans
|
|
||||||
|
|
||||||
def get_success_url(self, **kwargs):
|
|
||||||
self.object.refresh_from_db()
|
|
||||||
return reverse('food:food_view', kwargs={"pk": self.object.pk})
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
return context
|
|
||||||
|
|
||||||
|
|
||||||
class TransformedListView(ProtectQuerysetMixin, LoginRequiredMixin, MultiTableMixin, ListView):
|
|
||||||
"""
|
|
||||||
Displays ready TransformedFood
|
|
||||||
"""
|
|
||||||
model = TransformedFood
|
|
||||||
tables = [TransformedFoodTable, TransformedFoodTable, TransformedFoodTable]
|
|
||||||
extra_context = {"title": _("Transformed food")}
|
|
||||||
|
|
||||||
def get_queryset(self, **kwargs):
|
|
||||||
return super().get_queryset(**kwargs).distinct()
|
|
||||||
|
|
||||||
def get_tables(self):
|
|
||||||
tables = super().get_tables()
|
|
||||||
|
|
||||||
tables[0].prefix = "all-"
|
|
||||||
tables[1].prefix = "open-"
|
|
||||||
tables[2].prefix = "served-"
|
|
||||||
return tables
|
|
||||||
|
|
||||||
def get_tables_data(self):
|
|
||||||
# first table = all transformed food, second table = free, third = served
|
|
||||||
return [
|
|
||||||
self.get_queryset().order_by("-creation_date"),
|
|
||||||
TransformedFood.objects.filter(is_ready=True, is_active=True, was_eaten=False, expiry_date__lt=timezone.now())
|
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, TransformedFood, "view"))
|
|
||||||
.distinct()
|
|
||||||
.order_by("-creation_date"),
|
|
||||||
TransformedFood.objects.filter(is_ready=True, is_active=True, was_eaten=False, expiry_date__gte=timezone.now())
|
|
||||||
.filter(PermissionBackend.filter_queryset(self.request, TransformedFood, "view"))
|
|
||||||
.distinct()
|
|
||||||
.order_by("-creation_date")
|
|
||||||
]
|
|
||||||
|
|
||||||
def get_context_data(self, **kwargs):
|
|
||||||
context = super().get_context_data(**kwargs)
|
|
||||||
|
|
||||||
# We choose a club which should work
|
|
||||||
for membership in self.request.user.memberships.all():
|
|
||||||
club_id = membership.club.id
|
|
||||||
food = TransformedFood(
|
|
||||||
name="",
|
|
||||||
owner_id=club_id,
|
|
||||||
creation_date=timezone.now(),
|
|
||||||
expiry_date=timezone.now(),
|
|
||||||
)
|
|
||||||
if PermissionBackend.check_perm(self.request, "food.add_transformedfood", food):
|
|
||||||
context['can_create_meal'] = True
|
|
||||||
break
|
|
||||||
|
|
||||||
tables = context["tables"]
|
|
||||||
for name, table in zip(["table", "open", "served"], tables):
|
|
||||||
context[name] = table
|
|
||||||
return context
|
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
@ -76,6 +76,9 @@ class Changelog(models.Model):
|
|||||||
verbose_name=_('timestamp'),
|
verbose_name=_('timestamp'),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def delete(self, using=None, keep_parents=False):
|
||||||
|
raise ValidationError(_("Logs cannot be destroyed."))
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
verbose_name = _("changelog")
|
verbose_name = _("changelog")
|
||||||
verbose_name_plural = _("changelogs")
|
verbose_name_plural = _("changelogs")
|
||||||
@ -83,6 +86,3 @@ class Changelog(models.Model):
|
|||||||
def __str__(self):
|
def __str__(self):
|
||||||
return _("Changelog of type \"{action}\" for model {model} at {timestamp}").format(
|
return _("Changelog of type \"{action}\" for model {model} at {timestamp}").format(
|
||||||
action=self.get_action_display(), model=str(self.model), timestamp=str(self.timestamp))
|
action=self.get_action_display(), model=str(self.model), timestamp=str(self.timestamp))
|
||||||
|
|
||||||
def delete(self, using=None, keep_parents=False):
|
|
||||||
raise ValidationError(_("Logs cannot be destroyed."))
|
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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 rest_framework.renderers import JSONRenderer
|
from rest_framework.renderers import JSONRenderer
|
||||||
from rest_framework.serializers import ModelSerializer
|
from rest_framework.serializers import ModelSerializer
|
||||||
from note.models import NoteUser, Alias
|
from note.models import NoteUser, Alias
|
||||||
from note_kfet.middlewares import get_current_request
|
from note_kfet.middlewares import get_current_authenticated_user, get_current_ip
|
||||||
|
|
||||||
from .models import Changelog
|
from .models import Changelog
|
||||||
|
|
||||||
@ -56,13 +56,13 @@ def save_object(sender, instance, **kwargs):
|
|||||||
# noinspection PyProtectedMember
|
# noinspection PyProtectedMember
|
||||||
previous = instance._previous
|
previous = instance._previous
|
||||||
|
|
||||||
# Si un⋅e utilisateur⋅rice est connecté⋅e, on récupère l'utilisateur⋅rice courant⋅e ainsi que son adresse IP
|
# Si un utilisateur est connecté, on récupère l'utilisateur courant ainsi que son adresse IP
|
||||||
request = get_current_request()
|
user, ip = get_current_authenticated_user(), get_current_ip()
|
||||||
|
|
||||||
if request is None:
|
if user is None:
|
||||||
# Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py`
|
# Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py`
|
||||||
# On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée
|
# On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée
|
||||||
# IMPORTANT : l'utilisateur⋅rice dans la VM doit être un des alias note du respo info
|
# IMPORTANT : l'utilisateur dans la VM doit être un des alias note du respo info
|
||||||
ip = "127.0.0.1"
|
ip = "127.0.0.1"
|
||||||
username = Alias.normalize(getpass.getuser())
|
username = Alias.normalize(getpass.getuser())
|
||||||
note = NoteUser.objects.filter(alias__normalized_name=username)
|
note = NoteUser.objects.filter(alias__normalized_name=username)
|
||||||
@ -71,23 +71,9 @@ def save_object(sender, instance, **kwargs):
|
|||||||
# else:
|
# else:
|
||||||
if note.exists():
|
if note.exists():
|
||||||
user = note.get().user
|
user = note.get().user
|
||||||
else:
|
|
||||||
user = None
|
|
||||||
else:
|
|
||||||
user = request.user
|
|
||||||
if 'HTTP_X_REAL_IP' in request.META:
|
|
||||||
ip = request.META.get('HTTP_X_REAL_IP')
|
|
||||||
elif 'HTTP_X_FORWARDED_FOR' in request.META:
|
|
||||||
ip = request.META.get('HTTP_X_FORWARDED_FOR').split(', ')[0]
|
|
||||||
else:
|
|
||||||
ip = request.META.get('REMOTE_ADDR')
|
|
||||||
|
|
||||||
if not user.is_authenticated:
|
|
||||||
# For registration and OAuth2 purposes
|
|
||||||
user = None
|
|
||||||
|
|
||||||
# noinspection PyProtectedMember
|
# noinspection PyProtectedMember
|
||||||
if request is not None and instance._meta.label_lower == "auth.user" and previous:
|
if user is not None and instance._meta.label_lower == "auth.user" and previous:
|
||||||
# On n'enregistre pas les connexions
|
# On n'enregistre pas les connexions
|
||||||
if instance.last_login != previous.last_login:
|
if instance.last_login != previous.last_login:
|
||||||
return
|
return
|
||||||
@ -134,13 +120,13 @@ def delete_object(sender, instance, **kwargs):
|
|||||||
if instance._meta.label_lower in EXCLUDED or hasattr(instance, "_no_signal"):
|
if instance._meta.label_lower in EXCLUDED or hasattr(instance, "_no_signal"):
|
||||||
return
|
return
|
||||||
|
|
||||||
# Si un⋅e utilisateur⋅rice est connecté⋅e, on récupère l'utilisateur⋅rice courant⋅e ainsi que son adresse IP
|
# Si un utilisateur est connecté, on récupère l'utilisateur courant ainsi que son adresse IP
|
||||||
request = get_current_request()
|
user, ip = get_current_authenticated_user(), get_current_ip()
|
||||||
|
|
||||||
if request is None:
|
if user is None:
|
||||||
# Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py`
|
# Si la modification n'a pas été faite via le client Web, on suppose que c'est du à `manage.py`
|
||||||
# On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée
|
# On récupère alors l'utilisateur·trice connecté·e à la VM, et on récupère la note associée
|
||||||
# IMPORTANT : l'utilisateur⋅rice dans la VM doit être un des alias note du respo info
|
# IMPORTANT : l'utilisateur dans la VM doit être un des alias note du respo info
|
||||||
ip = "127.0.0.1"
|
ip = "127.0.0.1"
|
||||||
username = Alias.normalize(getpass.getuser())
|
username = Alias.normalize(getpass.getuser())
|
||||||
note = NoteUser.objects.filter(alias__normalized_name=username)
|
note = NoteUser.objects.filter(alias__normalized_name=username)
|
||||||
@ -149,20 +135,6 @@ def delete_object(sender, instance, **kwargs):
|
|||||||
# else:
|
# else:
|
||||||
if note.exists():
|
if note.exists():
|
||||||
user = note.get().user
|
user = note.get().user
|
||||||
else:
|
|
||||||
user = None
|
|
||||||
else:
|
|
||||||
user = request.user
|
|
||||||
if 'HTTP_X_REAL_IP' in request.META:
|
|
||||||
ip = request.META.get('HTTP_X_REAL_IP')
|
|
||||||
elif 'HTTP_X_FORWARDED_FOR' in request.META:
|
|
||||||
ip = request.META.get('HTTP_X_FORWARDED_FOR').split(', ')[0]
|
|
||||||
else:
|
|
||||||
ip = request.META.get('REMOTE_ADDR')
|
|
||||||
|
|
||||||
if not user.is_authenticated:
|
|
||||||
# For registration and OAuth2 purposes
|
|
||||||
user = None
|
|
||||||
|
|
||||||
# On crée notre propre sérialiseur JSON pour pouvoir sauvegarder les modèles
|
# On crée notre propre sérialiseur JSON pour pouvoir sauvegarder les modèles
|
||||||
class CustomSerializer(ModelSerializer):
|
class CustomSerializer(ModelSerializer):
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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,9 +1,8 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
from rest_framework.filters import OrderingFilter
|
from rest_framework.filters import OrderingFilter, SearchFilter
|
||||||
from api.filters import RegexSafeSearchFilter
|
|
||||||
from api.viewsets import ReadProtectedModelViewSet
|
from api.viewsets import ReadProtectedModelViewSet
|
||||||
|
|
||||||
from .serializers import ProfileSerializer, ClubSerializer, MembershipSerializer
|
from .serializers import ProfileSerializer, ClubSerializer, MembershipSerializer
|
||||||
@ -18,7 +17,7 @@ class ProfileViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Profile.objects.order_by('id')
|
queryset = Profile.objects.order_by('id')
|
||||||
serializer_class = ProfileSerializer
|
serializer_class = ProfileSerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['user', 'user__first_name', 'user__last_name', 'user__username', 'user__email',
|
filterset_fields = ['user', 'user__first_name', 'user__last_name', 'user__username', 'user__email',
|
||||||
'user__note__alias__name', 'user__note__alias__normalized_name', 'phone_number', "section",
|
'user__note__alias__name', 'user__note__alias__normalized_name', 'phone_number', "section",
|
||||||
'department', 'promotion', 'address', 'paid', 'ml_events_registration', 'ml_sport_registration',
|
'department', 'promotion', 'address', 'paid', 'ml_events_registration', 'ml_sport_registration',
|
||||||
@ -35,7 +34,7 @@ class ClubViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Club.objects.order_by('id')
|
queryset = Club.objects.order_by('id')
|
||||||
serializer_class = ClubSerializer
|
serializer_class = ClubSerializer
|
||||||
filter_backends = [DjangoFilterBackend, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, SearchFilter]
|
||||||
filterset_fields = ['name', 'email', 'note__alias__name', 'note__alias__normalized_name', 'parent_club',
|
filterset_fields = ['name', 'email', 'note__alias__name', 'note__alias__normalized_name', 'parent_club',
|
||||||
'parent_club__name', 'require_memberships', 'membership_fee_paid', 'membership_fee_unpaid',
|
'parent_club__name', 'require_memberships', 'membership_fee_paid', 'membership_fee_unpaid',
|
||||||
'membership_duration', 'membership_start', 'membership_end', ]
|
'membership_duration', 'membership_start', 'membership_end', ]
|
||||||
@ -50,7 +49,7 @@ class MembershipViewSet(ReadProtectedModelViewSet):
|
|||||||
"""
|
"""
|
||||||
queryset = Membership.objects.order_by('id')
|
queryset = Membership.objects.order_by('id')
|
||||||
serializer_class = MembershipSerializer
|
serializer_class = MembershipSerializer
|
||||||
filter_backends = [DjangoFilterBackend, OrderingFilter, RegexSafeSearchFilter]
|
filter_backends = [DjangoFilterBackend, OrderingFilter, SearchFilter]
|
||||||
filterset_fields = ['club__name', 'club__email', 'club__note__alias__name', 'club__note__alias__normalized_name',
|
filterset_fields = ['club__name', 'club__email', 'club__note__alias__name', 'club__note__alias__normalized_name',
|
||||||
'user__username', 'user__last_name', 'user__first_name', 'user__email',
|
'user__username', 'user__last_name', 'user__first_name', 'user__email',
|
||||||
'user__note__alias__name', 'user__note__alias__normalized_name',
|
'user__note__alias__name', 'user__note__alias__normalized_name',
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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,17 +0,0 @@
|
|||||||
# Copyright (C) 2018-2025 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,9 +1,9 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
|
|
||||||
from bootstrap_datepicker_plus.widgets import DatePickerInput
|
from PIL import Image, ImageSequence
|
||||||
from django import forms
|
from django import forms
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.auth.forms import AuthenticationForm
|
from django.contrib.auth.forms import AuthenticationForm
|
||||||
@ -13,9 +13,8 @@ from django.forms import CheckboxSelectMultiple
|
|||||||
from django.utils import timezone
|
from django.utils import timezone
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from note.models import NoteSpecial, Alias
|
from note.models import NoteSpecial, Alias
|
||||||
from note_kfet.inputs import Autocomplete, AmountInput
|
from note_kfet.inputs import Autocomplete, AmountInput, DatePickerInput
|
||||||
from permission.models import PermissionMask, Role
|
from permission.models import PermissionMask, Role
|
||||||
from PIL import Image, ImageSequence
|
|
||||||
|
|
||||||
from .models import Profile, Club, Membership
|
from .models import Profile, Club, Membership
|
||||||
|
|
||||||
@ -33,7 +32,7 @@ class UserForm(forms.ModelForm):
|
|||||||
# Django usernames can only contain letters, numbers, @, ., +, - and _.
|
# Django usernames can only contain letters, numbers, @, ., +, - and _.
|
||||||
# We want to allow users to have uncommon and unpractical usernames:
|
# We want to allow users to have uncommon and unpractical usernames:
|
||||||
# That is their problem, and we have normalized aliases for us.
|
# That is their problem, and we have normalized aliases for us.
|
||||||
return super()._get_validation_exclusions() | {"username"}
|
return super()._get_validation_exclusions() + ["username"]
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = User
|
model = User
|
||||||
@ -44,18 +43,10 @@ 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.
|
||||||
"""
|
"""
|
||||||
# Remove widget=forms.HiddenInput() if you want to use report frequency.
|
|
||||||
report_frequency = forms.IntegerField(required=False, initial=0, label=_("Report frequency"))
|
report_frequency = forms.IntegerField(required=False, initial=0, label=_("Report frequency"))
|
||||||
|
|
||||||
last_report = forms.DateTimeField(required=False, disabled=True, label=_("Last report date"))
|
last_report = forms.DateTimeField(required=False, disabled=True, label=_("Last report date"))
|
||||||
|
|
||||||
VSS_charter_read = forms.BooleanField(
|
|
||||||
required=True,
|
|
||||||
label=_("Anti-VSS (<em>Violences Sexistes et Sexuelles</em>) charter read and approved"),
|
|
||||||
help_text=_("Tick after having read and accepted the anti-VSS charter \
|
|
||||||
<a href=https://perso.crans.org/club-bde/Charte-anti-VSS.pdf target=_blank> available here in pdf</a>")
|
|
||||||
)
|
|
||||||
|
|
||||||
def clean_promotion(self):
|
def clean_promotion(self):
|
||||||
promotion = self.cleaned_data["promotion"]
|
promotion = self.cleaned_data["promotion"]
|
||||||
if promotion > timezone.now().year:
|
if promotion > timezone.now().year:
|
||||||
@ -77,8 +68,7 @@ class ProfileForm(forms.ModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = Profile
|
model = Profile
|
||||||
fields = '__all__'
|
fields = '__all__'
|
||||||
# Remove ml_[asso]_registration from exclude if the concerned association uses nk20 to manage its mailing list.
|
exclude = ('user', 'email_confirmed', 'registration_valid', )
|
||||||
exclude = ('user', 'email_confirmed', 'registration_valid', 'ml_sport_registration', )
|
|
||||||
|
|
||||||
|
|
||||||
class ImageForm(forms.Form):
|
class ImageForm(forms.Form):
|
||||||
@ -124,7 +114,7 @@ class ImageForm(forms.Form):
|
|||||||
frame = frame.crop((x, y, x + w, y + h))
|
frame = frame.crop((x, y, x + w, y + h))
|
||||||
frame = frame.resize(
|
frame = frame.resize(
|
||||||
(settings.PIC_WIDTH, settings.PIC_RATIO * settings.PIC_WIDTH),
|
(settings.PIC_WIDTH, settings.PIC_RATIO * settings.PIC_WIDTH),
|
||||||
Image.LANCZOS,
|
Image.ANTIALIAS,
|
||||||
)
|
)
|
||||||
frames.append(frame)
|
frames.append(frame)
|
||||||
|
|
||||||
@ -141,9 +131,6 @@ class ImageForm(forms.Form):
|
|||||||
|
|
||||||
return cleaned_data
|
return cleaned_data
|
||||||
|
|
||||||
def is_valid(self):
|
|
||||||
return super().is_valid() or super().clean().get('image') is None
|
|
||||||
|
|
||||||
|
|
||||||
class ClubForm(forms.ModelForm):
|
class ClubForm(forms.ModelForm):
|
||||||
def clean(self):
|
def clean(self):
|
||||||
@ -157,7 +144,7 @@ class ClubForm(forms.ModelForm):
|
|||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = Club
|
model = Club
|
||||||
exclude = ("add_registration_form",)
|
fields = '__all__'
|
||||||
widgets = {
|
widgets = {
|
||||||
"membership_fee_paid": AmountInput(),
|
"membership_fee_paid": AmountInput(),
|
||||||
"membership_fee_unpaid": AmountInput(),
|
"membership_fee_unpaid": AmountInput(),
|
||||||
@ -213,9 +200,9 @@ class MembershipForm(forms.ModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = Membership
|
model = Membership
|
||||||
fields = ('user', 'date_start')
|
fields = ('user', 'date_start')
|
||||||
# Le champ d'utilisateur⋅rice est remplacé par un champ d'auto-complétion.
|
# Le champ d'utilisateur est remplacé par un champ d'auto-complétion.
|
||||||
# Quand des lettres sont tapées, une requête est envoyée sur l'API d'auto-complétion
|
# Quand des lettres sont tapées, une requête est envoyée sur l'API d'auto-complétion
|
||||||
# et récupère les noms d'utilisateur⋅rices valides
|
# et récupère les noms d'utilisateur valides
|
||||||
widgets = {
|
widgets = {
|
||||||
'user':
|
'user':
|
||||||
Autocomplete(
|
Autocomplete(
|
||||||
|
@ -1,14 +1,12 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
from collections import OrderedDict
|
|
||||||
|
|
||||||
from django.conf import settings
|
from django.conf import settings
|
||||||
from django.contrib.auth.hashers import PBKDF2PasswordHasher, mask_hash
|
from django.contrib.auth.hashers import PBKDF2PasswordHasher
|
||||||
from django.utils.crypto import constant_time_compare
|
from django.utils.crypto import constant_time_compare
|
||||||
from django.utils.translation import gettext_lazy as _
|
from note_kfet.middlewares import get_current_authenticated_user, get_current_session
|
||||||
from note_kfet.middlewares import get_current_request
|
|
||||||
|
|
||||||
|
|
||||||
class CustomNK15Hasher(PBKDF2PasswordHasher):
|
class CustomNK15Hasher(PBKDF2PasswordHasher):
|
||||||
@ -26,22 +24,16 @@ class CustomNK15Hasher(PBKDF2PasswordHasher):
|
|||||||
|
|
||||||
def must_update(self, encoded):
|
def must_update(self, encoded):
|
||||||
if settings.DEBUG:
|
if settings.DEBUG:
|
||||||
# Small hack to let superusers to impersonate people.
|
current_user = get_current_authenticated_user()
|
||||||
# Don't change their password.
|
|
||||||
request = get_current_request()
|
|
||||||
current_user = request.user
|
|
||||||
if current_user is not None and current_user.is_superuser:
|
if current_user is not None and current_user.is_superuser:
|
||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def verify(self, password, encoded):
|
def verify(self, password, encoded):
|
||||||
if settings.DEBUG:
|
if settings.DEBUG:
|
||||||
# Small hack to let superusers to impersonate people.
|
current_user = get_current_authenticated_user()
|
||||||
# If a superuser is already connected, let him/her log in as another person.
|
|
||||||
request = get_current_request()
|
|
||||||
current_user = request.user
|
|
||||||
if current_user is not None and current_user.is_superuser\
|
if current_user is not None and current_user.is_superuser\
|
||||||
and request.session.get("permission_mask", -1) >= 42:
|
and get_current_session().get("permission_mask", -1) >= 42:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
if '|' in encoded:
|
if '|' in encoded:
|
||||||
@ -49,18 +41,6 @@ class CustomNK15Hasher(PBKDF2PasswordHasher):
|
|||||||
return constant_time_compare(hashlib.sha256((salt + password).encode("utf-8")).hexdigest(), db_hashed_pass)
|
return constant_time_compare(hashlib.sha256((salt + password).encode("utf-8")).hexdigest(), db_hashed_pass)
|
||||||
return super().verify(password, encoded)
|
return super().verify(password, encoded)
|
||||||
|
|
||||||
def safe_summary(self, encoded):
|
|
||||||
# Displayed information in Django Admin.
|
|
||||||
if '|' in encoded:
|
|
||||||
salt, db_hashed_pass = encoded.split('$')[2].split('|')
|
|
||||||
return OrderedDict([
|
|
||||||
(_('algorithm'), 'custom_nk15'),
|
|
||||||
(_('iterations'), '1'),
|
|
||||||
(_('salt'), mask_hash(salt)),
|
|
||||||
(_('hash'), mask_hash(db_hashed_pass)),
|
|
||||||
])
|
|
||||||
return super().safe_summary(encoded)
|
|
||||||
|
|
||||||
|
|
||||||
class DebugSuperuserBackdoor(PBKDF2PasswordHasher):
|
class DebugSuperuserBackdoor(PBKDF2PasswordHasher):
|
||||||
"""
|
"""
|
||||||
@ -71,11 +51,8 @@ class DebugSuperuserBackdoor(PBKDF2PasswordHasher):
|
|||||||
|
|
||||||
def verify(self, password, encoded):
|
def verify(self, password, encoded):
|
||||||
if settings.DEBUG:
|
if settings.DEBUG:
|
||||||
# Small hack to let superusers to impersonate people.
|
current_user = get_current_authenticated_user()
|
||||||
# If a superuser is already connected, let him/her log in as another person.
|
|
||||||
request = get_current_request()
|
|
||||||
current_user = request.user
|
|
||||||
if current_user is not None and current_user.is_superuser\
|
if current_user is not None and current_user.is_superuser\
|
||||||
and request.session.get("permission_mask", -1) >= 42:
|
and get_current_session().get("permission_mask", -1) >= 42:
|
||||||
return True
|
return True
|
||||||
return super().verify(password, encoded)
|
return super().verify(password, encoded)
|
||||||
|
@ -19,8 +19,8 @@ def create_bde_and_kfet(apps, schema_editor):
|
|||||||
membership_fee_paid=500,
|
membership_fee_paid=500,
|
||||||
membership_fee_unpaid=500,
|
membership_fee_unpaid=500,
|
||||||
membership_duration=396,
|
membership_duration=396,
|
||||||
membership_start="2021-08-01",
|
membership_start="2020-08-01",
|
||||||
membership_end="2022-09-30",
|
membership_end="2021-09-30",
|
||||||
)
|
)
|
||||||
Club.objects.get_or_create(
|
Club.objects.get_or_create(
|
||||||
id=2,
|
id=2,
|
||||||
@ -31,8 +31,8 @@ def create_bde_and_kfet(apps, schema_editor):
|
|||||||
membership_fee_paid=3500,
|
membership_fee_paid=3500,
|
||||||
membership_fee_unpaid=3500,
|
membership_fee_unpaid=3500,
|
||||||
membership_duration=396,
|
membership_duration=396,
|
||||||
membership_start="2021-08-01",
|
membership_start="2020-08-01",
|
||||||
membership_end="2022-09-30",
|
membership_end="2021-09-30",
|
||||||
)
|
)
|
||||||
|
|
||||||
NoteClub.objects.get_or_create(
|
NoteClub.objects.get_or_create(
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
# 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,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.24 on 2021-10-05 13:44
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0007_auto_20210313_1235'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='profile',
|
|
||||||
name='department',
|
|
||||||
field=models.CharField(choices=[('A0', 'Informatics (A0)'), ('A1', 'Mathematics (A1)'), ('A2', 'Physics (A2)'), ("A'2", "Applied physics (A'2)"), ("A''2", "Chemistry (A''2)"), ('A3', 'Biology (A3)'), ('B1234', 'SAPHIRE (B1234)'), ('B1', 'Mechanics (B1)'), ('B2', 'Civil engineering (B2)'), ('B3', 'Mechanical engineering (B3)'), ('B4', 'EEA (B4)'), ('C', 'Design (C)'), ('D2', 'Economy-management (D2)'), ('D3', 'Social sciences (D3)'), ('E', 'English (E)'), ('EXT', 'External (EXT)')], max_length=8, verbose_name='department'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.26 on 2022-09-04 21:25
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0008_auto_20211005_1544'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='profile',
|
|
||||||
name='promotion',
|
|
||||||
field=models.PositiveSmallIntegerField(default=2022, help_text='Year of entry to the school (None if not ENS student)', null=True, verbose_name='promotion'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2023-08-23 21:29
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0009_auto_20220904_2325'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='profile',
|
|
||||||
name='promotion',
|
|
||||||
field=models.PositiveSmallIntegerField(default=2023, help_text='Year of entry to the school (None if not ENS student)', null=True, verbose_name='promotion'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2023-08-31 09:50
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0010_new_default_year'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name='profile',
|
|
||||||
name='VSS_charter_read',
|
|
||||||
field=models.BooleanField(default=False, verbose_name='VSS charter read'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-07-15 09:24
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0011_profile_vss_charter_read'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AddField(
|
|
||||||
model_name='club',
|
|
||||||
name='add_registration_form',
|
|
||||||
field=models.BooleanField(default=False, verbose_name='add to registration form'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,18 +0,0 @@
|
|||||||
# Generated by Django 2.2.28 on 2024-08-01 12:36
|
|
||||||
|
|
||||||
from django.db import migrations, models
|
|
||||||
|
|
||||||
|
|
||||||
class Migration(migrations.Migration):
|
|
||||||
|
|
||||||
dependencies = [
|
|
||||||
('member', '0012_club_add_registration_form'),
|
|
||||||
]
|
|
||||||
|
|
||||||
operations = [
|
|
||||||
migrations.AlterField(
|
|
||||||
model_name='profile',
|
|
||||||
name='promotion',
|
|
||||||
field=models.PositiveSmallIntegerField(default=2024, help_text='Year of entry to the school (None if not ENS student)', null=True, verbose_name='promotion'),
|
|
||||||
),
|
|
||||||
]
|
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
@ -28,6 +28,7 @@ class Profile(models.Model):
|
|||||||
We do not want to patch the Django Contrib :model:`auth.User`model;
|
We do not want to patch the Django Contrib :model:`auth.User`model;
|
||||||
so this model add an user profile with additional information.
|
so this model add an user profile with additional information.
|
||||||
"""
|
"""
|
||||||
|
|
||||||
user = models.OneToOneField(
|
user = models.OneToOneField(
|
||||||
settings.AUTH_USER_MODEL,
|
settings.AUTH_USER_MODEL,
|
||||||
on_delete=models.CASCADE,
|
on_delete=models.CASCADE,
|
||||||
@ -56,7 +57,7 @@ class Profile(models.Model):
|
|||||||
('A1', _("Mathematics (A1)")),
|
('A1', _("Mathematics (A1)")),
|
||||||
('A2', _("Physics (A2)")),
|
('A2', _("Physics (A2)")),
|
||||||
("A'2", _("Applied physics (A'2)")),
|
("A'2", _("Applied physics (A'2)")),
|
||||||
("A''2", _("Chemistry (A''2)")),
|
('A''2', _("Chemistry (A''2)")),
|
||||||
('A3', _("Biology (A3)")),
|
('A3', _("Biology (A3)")),
|
||||||
('B1234', _("SAPHIRE (B1234)")),
|
('B1234', _("SAPHIRE (B1234)")),
|
||||||
('B1', _("Mechanics (B1)")),
|
('B1', _("Mechanics (B1)")),
|
||||||
@ -73,7 +74,7 @@ class Profile(models.Model):
|
|||||||
|
|
||||||
promotion = models.PositiveSmallIntegerField(
|
promotion = models.PositiveSmallIntegerField(
|
||||||
null=True,
|
null=True,
|
||||||
default=datetime.date.today().year if datetime.date.today().month >= 8 else datetime.date.today().year - 1,
|
default=datetime.date.today().year,
|
||||||
verbose_name=_("promotion"),
|
verbose_name=_("promotion"),
|
||||||
help_text=_("Year of entry to the school (None if not ENS student)"),
|
help_text=_("Year of entry to the school (None if not ENS student)"),
|
||||||
)
|
)
|
||||||
@ -133,22 +134,6 @@ class Profile(models.Model):
|
|||||||
default=False,
|
default=False,
|
||||||
)
|
)
|
||||||
|
|
||||||
VSS_charter_read = models.BooleanField(
|
|
||||||
verbose_name=_("VSS charter read"),
|
|
||||||
default=False
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('user profile')
|
|
||||||
verbose_name_plural = _('user profile')
|
|
||||||
indexes = [models.Index(fields=['user'])]
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return str(self.user)
|
|
||||||
|
|
||||||
def get_absolute_url(self):
|
|
||||||
return reverse('member:user_detail', args=(self.user_id,))
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def ens_year(self):
|
def ens_year(self):
|
||||||
"""
|
"""
|
||||||
@ -173,6 +158,17 @@ class Profile(models.Model):
|
|||||||
return SogeCredit.objects.filter(user=self.user, credit_transaction__isnull=False).exists()
|
return SogeCredit.objects.filter(user=self.user, credit_transaction__isnull=False).exists()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _('user profile')
|
||||||
|
verbose_name_plural = _('user profile')
|
||||||
|
indexes = [models.Index(fields=['user'])]
|
||||||
|
|
||||||
|
def get_absolute_url(self):
|
||||||
|
return reverse('member:user_detail', args=(self.user_id,))
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return str(self.user)
|
||||||
|
|
||||||
def send_email_validation_link(self):
|
def send_email_validation_link(self):
|
||||||
subject = "[Note Kfet] " + str(_("Activate your Note Kfet account"))
|
subject = "[Note Kfet] " + str(_("Activate your Note Kfet account"))
|
||||||
token = email_validation_token.make_token(self.user)
|
token = email_validation_token.make_token(self.user)
|
||||||
@ -204,11 +200,9 @@ class Club(models.Model):
|
|||||||
max_length=255,
|
max_length=255,
|
||||||
unique=True,
|
unique=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
email = models.EmailField(
|
email = models.EmailField(
|
||||||
verbose_name=_('email'),
|
verbose_name=_('email'),
|
||||||
)
|
)
|
||||||
|
|
||||||
parent_club = models.ForeignKey(
|
parent_club = models.ForeignKey(
|
||||||
'self',
|
'self',
|
||||||
null=True,
|
null=True,
|
||||||
@ -259,17 +253,23 @@ class Club(models.Model):
|
|||||||
help_text=_('Maximal date of a membership, after which members must renew it.'),
|
help_text=_('Maximal date of a membership, after which members must renew it.'),
|
||||||
)
|
)
|
||||||
|
|
||||||
add_registration_form = models.BooleanField(
|
def update_membership_dates(self):
|
||||||
verbose_name=_("add to registration form"),
|
"""
|
||||||
default=False,
|
This function is called each time the club detail view is displayed.
|
||||||
)
|
Update the year of the membership dates.
|
||||||
|
"""
|
||||||
|
if not self.membership_start:
|
||||||
|
return
|
||||||
|
|
||||||
class Meta:
|
today = datetime.date.today()
|
||||||
verbose_name = _("club")
|
|
||||||
verbose_name_plural = _("clubs")
|
|
||||||
|
|
||||||
def __str__(self):
|
if (today - self.membership_start).days >= 365:
|
||||||
return self.name
|
self.membership_start = datetime.date(self.membership_start.year + 1,
|
||||||
|
self.membership_start.month, self.membership_start.day)
|
||||||
|
self.membership_end = datetime.date(self.membership_end.year + 1,
|
||||||
|
self.membership_end.month, self.membership_end.day)
|
||||||
|
self._force_save = True
|
||||||
|
self.save(force_update=True)
|
||||||
|
|
||||||
@transaction.atomic
|
@transaction.atomic
|
||||||
def save(self, force_insert=False, force_update=False, using=None,
|
def save(self, force_insert=False, force_update=False, using=None,
|
||||||
@ -282,36 +282,16 @@ class Club(models.Model):
|
|||||||
self.membership_end = None
|
self.membership_end = None
|
||||||
super().save(force_insert, force_update, update_fields)
|
super().save(force_insert, force_update, update_fields)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("club")
|
||||||
|
verbose_name_plural = _("clubs")
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
def get_absolute_url(self):
|
def get_absolute_url(self):
|
||||||
return reverse_lazy('member:club_detail', args=(self.pk,))
|
return reverse_lazy('member:club_detail', args=(self.pk,))
|
||||||
|
|
||||||
def update_membership_dates(self):
|
|
||||||
"""
|
|
||||||
This function is called each time the club detail view is displayed.
|
|
||||||
Update the year of the membership dates.
|
|
||||||
"""
|
|
||||||
if not self.membership_start or not self.membership_end:
|
|
||||||
return
|
|
||||||
|
|
||||||
today = datetime.date.today()
|
|
||||||
|
|
||||||
# Avoid any problems on February 29
|
|
||||||
if self.membership_start.month == 2 and self.membership_start.day == 29:
|
|
||||||
self.membership_start -= datetime.timedelta(days=1)
|
|
||||||
if self.membership_end.month == 2 and self.membership_end.day == 29:
|
|
||||||
self.membership_end += datetime.timedelta(days=1)
|
|
||||||
|
|
||||||
while today >= datetime.date(self.membership_start.year + 1,
|
|
||||||
self.membership_start.month, self.membership_start.day):
|
|
||||||
if self.membership_start:
|
|
||||||
self.membership_start = datetime.date(self.membership_start.year + 1,
|
|
||||||
self.membership_start.month, self.membership_start.day)
|
|
||||||
if self.membership_end:
|
|
||||||
self.membership_end = datetime.date(self.membership_end.year + 1,
|
|
||||||
self.membership_end.month, self.membership_end.day)
|
|
||||||
self._force_save = True
|
|
||||||
self.save(force_update=True)
|
|
||||||
|
|
||||||
|
|
||||||
class Membership(models.Model):
|
class Membership(models.Model):
|
||||||
"""
|
"""
|
||||||
@ -351,66 +331,6 @@ class Membership(models.Model):
|
|||||||
verbose_name=_('fee'),
|
verbose_name=_('fee'),
|
||||||
)
|
)
|
||||||
|
|
||||||
class Meta:
|
|
||||||
verbose_name = _('membership')
|
|
||||||
verbose_name_plural = _('memberships')
|
|
||||||
indexes = [models.Index(fields=['user'])]
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return _("Membership of {user} for the club {club}").format(user=self.user.username, club=self.club.name, )
|
|
||||||
|
|
||||||
@transaction.atomic
|
|
||||||
def save(self, *args, **kwargs):
|
|
||||||
"""
|
|
||||||
Calculate fee and end date before saving the membership and creating the transaction if needed.
|
|
||||||
"""
|
|
||||||
# Ensure that club membership dates are valid
|
|
||||||
old_membership_start = self.club.membership_start
|
|
||||||
self.club.update_membership_dates()
|
|
||||||
if self.club.membership_start != old_membership_start:
|
|
||||||
self.club.save()
|
|
||||||
|
|
||||||
created = not self.pk
|
|
||||||
if not created:
|
|
||||||
for role in self.roles.all():
|
|
||||||
club = role.for_club
|
|
||||||
if club is not None:
|
|
||||||
if club.pk != self.club_id:
|
|
||||||
raise ValidationError(_('The role {role} does not apply to the club {club}.')
|
|
||||||
.format(role=role.name, club=club.name))
|
|
||||||
else:
|
|
||||||
if Membership.objects.filter(
|
|
||||||
user=self.user,
|
|
||||||
club=self.club,
|
|
||||||
date_start__lte=self.date_start,
|
|
||||||
date_end__gte=self.date_start,
|
|
||||||
).exists():
|
|
||||||
raise ValidationError(_('User is already a member of the club'))
|
|
||||||
|
|
||||||
if self.club.parent_club is not None:
|
|
||||||
# Check that the user is already a member of the parent club if the membership is created
|
|
||||||
if not Membership.objects.filter(
|
|
||||||
user=self.user,
|
|
||||||
club=self.club.parent_club,
|
|
||||||
date_start__gte=self.club.parent_club.membership_start,
|
|
||||||
).exists():
|
|
||||||
if hasattr(self, '_force_renew_parent') and self._force_renew_parent:
|
|
||||||
self.renew_parent()
|
|
||||||
else:
|
|
||||||
raise ValidationError(_('User is not a member of the parent club')
|
|
||||||
+ ' ' + self.club.parent_club.name)
|
|
||||||
|
|
||||||
self.fee = self.club.membership_fee_paid if self.user.profile.paid else self.club.membership_fee_unpaid
|
|
||||||
|
|
||||||
self.date_end = self.date_start + datetime.timedelta(days=self.club.membership_duration) \
|
|
||||||
if self.club.membership_duration is not None else self.date_start + datetime.timedelta(days=424242)
|
|
||||||
if self.club.membership_end is not None and self.date_end > self.club.membership_end:
|
|
||||||
self.date_end = self.club.membership_end
|
|
||||||
|
|
||||||
super().save(*args, **kwargs)
|
|
||||||
|
|
||||||
self.make_transaction()
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def valid(self):
|
def valid(self):
|
||||||
"""
|
"""
|
||||||
@ -480,14 +400,60 @@ class Membership(models.Model):
|
|||||||
|
|
||||||
if self.club.parent_club.name == "BDE":
|
if self.club.parent_club.name == "BDE":
|
||||||
parent_membership.roles.set(
|
parent_membership.roles.set(
|
||||||
Role.objects.filter(Q(name="Adhérent⋅e BDE") | Q(name="Membre de club")).all())
|
Role.objects.filter(Q(name="Adhérent BDE") | Q(name="Membre de club")).all())
|
||||||
elif self.club.parent_club.name == "Kfet":
|
elif self.club.parent_club.name == "Kfet":
|
||||||
parent_membership.roles.set(
|
parent_membership.roles.set(
|
||||||
Role.objects.filter(Q(name="Adhérent⋅e Kfet") | Q(name="Membre de club")).all())
|
Role.objects.filter(Q(name="Adhérent Kfet") | Q(name="Membre de club")).all())
|
||||||
else:
|
else:
|
||||||
parent_membership.roles.set(Role.objects.filter(name="Membre de club").all())
|
parent_membership.roles.set(Role.objects.filter(name="Membre de club").all())
|
||||||
parent_membership.save()
|
parent_membership.save()
|
||||||
|
|
||||||
|
@transaction.atomic
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
Calculate fee and end date before saving the membership and creating the transaction if needed.
|
||||||
|
"""
|
||||||
|
created = not self.pk
|
||||||
|
if not created:
|
||||||
|
for role in self.roles.all():
|
||||||
|
club = role.for_club
|
||||||
|
if club is not None:
|
||||||
|
if club.pk != self.club_id:
|
||||||
|
raise ValidationError(_('The role {role} does not apply to the club {club}.')
|
||||||
|
.format(role=role.name, club=club.name))
|
||||||
|
else:
|
||||||
|
if Membership.objects.filter(
|
||||||
|
user=self.user,
|
||||||
|
club=self.club,
|
||||||
|
date_start__lte=self.date_start,
|
||||||
|
date_end__gte=self.date_start,
|
||||||
|
).exists():
|
||||||
|
raise ValidationError(_('User is already a member of the club'))
|
||||||
|
|
||||||
|
if self.club.parent_club is not None:
|
||||||
|
# Check that the user is already a member of the parent club if the membership is created
|
||||||
|
if not Membership.objects.filter(
|
||||||
|
user=self.user,
|
||||||
|
club=self.club.parent_club,
|
||||||
|
date_start__gte=self.club.parent_club.membership_start,
|
||||||
|
).exists():
|
||||||
|
if hasattr(self, '_force_renew_parent') and self._force_renew_parent:
|
||||||
|
self.renew_parent()
|
||||||
|
else:
|
||||||
|
raise ValidationError(_('User is not a member of the parent club')
|
||||||
|
+ ' ' + self.club.parent_club.name)
|
||||||
|
|
||||||
|
self.fee = self.club.membership_fee_paid if self.user.profile.paid else self.club.membership_fee_unpaid
|
||||||
|
|
||||||
|
self.date_end = self.date_start + datetime.timedelta(days=self.club.membership_duration) \
|
||||||
|
if self.club.membership_duration is not None else self.date_start + datetime.timedelta(days=424242)
|
||||||
|
if self.club.membership_end is not None and self.date_end > self.club.membership_end:
|
||||||
|
self.date_end = self.club.membership_end
|
||||||
|
|
||||||
|
super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
self.make_transaction()
|
||||||
|
|
||||||
def make_transaction(self):
|
def make_transaction(self):
|
||||||
"""
|
"""
|
||||||
Create Membership transaction associated to this membership.
|
Create Membership transaction associated to this membership.
|
||||||
@ -525,3 +491,11 @@ class Membership(models.Model):
|
|||||||
soge_credit.save()
|
soge_credit.save()
|
||||||
else:
|
else:
|
||||||
transaction.save(force_insert=True)
|
transaction.save(force_insert=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return _("Membership of {user} for the club {club}").format(user=self.user.username, club=self.club.name, )
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _('membership')
|
||||||
|
verbose_name_plural = _('memberships')
|
||||||
|
indexes = [models.Index(fields=['user'])]
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 by BDE ENS Paris-Saclay
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,64 +0,0 @@
|
|||||||
/**
|
|
||||||
* On form submit, create a new friendship
|
|
||||||
*/
|
|
||||||
function form_create_trust (e) {
|
|
||||||
// Do not submit HTML form
|
|
||||||
e.preventDefault()
|
|
||||||
|
|
||||||
// Get data and send to API
|
|
||||||
const formData = new FormData(e.target)
|
|
||||||
$.getJSON('/api/note/alias/'+formData.get('trusted') + '/',
|
|
||||||
function (trusted_alias) {
|
|
||||||
if ((trusted_alias.note == formData.get('trusting')))
|
|
||||||
{
|
|
||||||
addMsg(gettext("You can't add yourself as a friend"), "danger")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
create_trust(formData.get('trusting'), trusted_alias.note)
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a trust between users
|
|
||||||
* @param trusting:Integer trusting note id
|
|
||||||
* @param trusted:Integer trusted note id
|
|
||||||
*/
|
|
||||||
function create_trust(trusting, trusted) {
|
|
||||||
$.post('/api/note/trust/', {
|
|
||||||
trusting: trusting,
|
|
||||||
trusted: trusted,
|
|
||||||
csrfmiddlewaretoken: CSRF_TOKEN
|
|
||||||
}).done(function () {
|
|
||||||
// Reload tables
|
|
||||||
$('#trust_table').load(location.pathname + ' #trust_table')
|
|
||||||
$('#trusted_table').load(location.pathname + ' #trusted_table')
|
|
||||||
addMsg(gettext('Friendship successfully added'), 'success')
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* On click of "delete", delete the trust
|
|
||||||
* @param button_id:Integer Trust id to remove
|
|
||||||
*/
|
|
||||||
function delete_button (button_id) {
|
|
||||||
$.ajax({
|
|
||||||
url: '/api/note/trust/' + button_id + '/',
|
|
||||||
method: 'DELETE',
|
|
||||||
headers: { 'X-CSRFTOKEN': CSRF_TOKEN }
|
|
||||||
}).done(function () {
|
|
||||||
addMsg(gettext('Friendship successfully deleted'), 'success')
|
|
||||||
$('#trust_table').load(location.pathname + ' #trust_table')
|
|
||||||
$('#trusted_table').load(location.pathname + ' #trusted_table')
|
|
||||||
}).fail(function (xhr, _textStatus, _error) {
|
|
||||||
errMsg(xhr.responseJSON)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
$(document).ready(function () {
|
|
||||||
// Attach event
|
|
||||||
document.getElementById('form_trust').addEventListener('submit', form_create_trust)
|
|
||||||
})
|
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright (C) 2018-2025 by BDE ENS Paris-Saclay
|
# Copyright (C) 2018-2020 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
|
||||||
@ -9,7 +9,7 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from django.urls import reverse_lazy
|
from django.urls import reverse_lazy
|
||||||
from django.utils.html import format_html
|
from django.utils.html import format_html
|
||||||
from note.templatetags.pretty_money import pretty_money
|
from note.templatetags.pretty_money import pretty_money
|
||||||
from note_kfet.middlewares import get_current_request
|
from note_kfet.middlewares import get_current_authenticated_user
|
||||||
from permission.backends import PermissionBackend
|
from permission.backends import PermissionBackend
|
||||||
|
|
||||||
from .models import Club, Membership
|
from .models import Club, Membership
|
||||||
@ -31,8 +31,7 @@ class ClubTable(tables.Table):
|
|||||||
row_attrs = {
|
row_attrs = {
|
||||||
'class': 'table-row',
|
'class': 'table-row',
|
||||||
'id': lambda record: "row-" + str(record.pk),
|
'id': lambda record: "row-" + str(record.pk),
|
||||||
'data-href': lambda record: record.pk,
|
'data-href': lambda record: record.pk
|
||||||
'style': 'cursor:pointer',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -42,29 +41,29 @@ class UserTable(tables.Table):
|
|||||||
"""
|
"""
|
||||||
alias = tables.Column()
|
alias = tables.Column()
|
||||||
|
|
||||||
section = tables.Column(accessor='profile__section', orderable=False)
|
section = tables.Column(accessor='profile__section')
|
||||||
|
|
||||||
# Override the column to let replace the URL
|
# Override the column to let replace the URL
|
||||||
email = tables.EmailColumn(linkify=lambda record: "mailto:{}".format(record.email))
|
email = tables.EmailColumn(linkify=lambda record: "mailto:{}".format(record.email))
|
||||||
|
|
||||||
balance = tables.Column(accessor='note__balance', verbose_name=_("Balance"), orderable=False)
|
balance = tables.Column(accessor='note__balance', verbose_name=_("Balance"))
|
||||||
|
|
||||||
def render_email(self, record, value):
|
def render_email(self, record, value):
|
||||||
# Replace the email by a dash if the user can't see the profile detail
|
# Replace the email by a dash if the user can't see the profile detail
|
||||||
# Replace also the URL
|
# Replace also the URL
|
||||||
if not PermissionBackend.check_perm(get_current_request(), "member.view_profile", record.profile):
|
if not PermissionBackend.check_perm(get_current_authenticated_user(), "member.view_profile", record.profile):
|
||||||
value = "—"
|
value = "—"
|
||||||
record.email = value
|
record.email = value
|
||||||
return value
|
return value
|
||||||
|
|
||||||
def render_section(self, record, value):
|
def render_section(self, record, value):
|
||||||
return value \
|
return value \
|
||||||
if PermissionBackend.check_perm(get_current_request(), "member.view_profile", record.profile) \
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "member.view_profile", record.profile) \
|
||||||
else "—"
|
else "—"
|
||||||
|
|
||||||
def render_balance(self, record, value):
|
def render_balance(self, record, value):
|
||||||
return pretty_money(value)\
|
return pretty_money(value)\
|
||||||
if PermissionBackend.check_perm(get_current_request(), "note.view_note", record.note) else "—"
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "note.view_note", record.note) else "—"
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
attrs = {
|
attrs = {
|
||||||
@ -75,8 +74,7 @@ class UserTable(tables.Table):
|
|||||||
model = User
|
model = User
|
||||||
row_attrs = {
|
row_attrs = {
|
||||||
'class': 'table-row',
|
'class': 'table-row',
|
||||||
'data-href': lambda record: record.pk,
|
'data-href': lambda record: record.pk
|
||||||
'style': 'cursor:pointer',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -95,7 +93,7 @@ class MembershipTable(tables.Table):
|
|||||||
def render_user(self, value):
|
def render_user(self, value):
|
||||||
# If the user has the right, link the displayed user with the page of its detail.
|
# If the user has the right, link the displayed user with the page of its detail.
|
||||||
s = value.username
|
s = value.username
|
||||||
if PermissionBackend.check_perm(get_current_request(), "auth.view_user", value):
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "auth.view_user", value):
|
||||||
s = format_html("<a href={url}>{name}</a>",
|
s = format_html("<a href={url}>{name}</a>",
|
||||||
url=reverse_lazy('member:user_detail', kwargs={"pk": value.pk}), name=s)
|
url=reverse_lazy('member:user_detail', kwargs={"pk": value.pk}), name=s)
|
||||||
|
|
||||||
@ -104,7 +102,7 @@ class MembershipTable(tables.Table):
|
|||||||
def render_club(self, value):
|
def render_club(self, value):
|
||||||
# If the user has the right, link the displayed club with the page of its detail.
|
# If the user has the right, link the displayed club with the page of its detail.
|
||||||
s = value.name
|
s = value.name
|
||||||
if PermissionBackend.check_perm(get_current_request(), "member.view_club", value):
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "member.view_club", value):
|
||||||
s = format_html("<a href={url}>{name}</a>",
|
s = format_html("<a href={url}>{name}</a>",
|
||||||
url=reverse_lazy('member:club_detail', kwargs={"pk": value.pk}), name=s)
|
url=reverse_lazy('member:club_detail', kwargs={"pk": value.pk}), name=s)
|
||||||
|
|
||||||
@ -120,7 +118,7 @@ class MembershipTable(tables.Table):
|
|||||||
club=record.club,
|
club=record.club,
|
||||||
user=record.user,
|
user=record.user,
|
||||||
date_start__gte=record.club.membership_start,
|
date_start__gte=record.club.membership_start,
|
||||||
date_end__lte=record.club.membership_end or date(9999, 12, 31),
|
date_end__lte=record.club.membership_end,
|
||||||
).exists(): # If the renew is not yet performed
|
).exists(): # If the renew is not yet performed
|
||||||
empty_membership = Membership(
|
empty_membership = Membership(
|
||||||
club=record.club,
|
club=record.club,
|
||||||
@ -129,7 +127,7 @@ class MembershipTable(tables.Table):
|
|||||||
date_end=date.today(),
|
date_end=date.today(),
|
||||||
fee=0,
|
fee=0,
|
||||||
)
|
)
|
||||||
if PermissionBackend.check_perm(get_current_request(),
|
if PermissionBackend.check_perm(get_current_authenticated_user(),
|
||||||
"member.add_membership", empty_membership): # If the user has right
|
"member.add_membership", empty_membership): # If the user has right
|
||||||
renew_url = reverse_lazy('member:club_renew_membership',
|
renew_url = reverse_lazy('member:club_renew_membership',
|
||||||
kwargs={"pk": record.pk})
|
kwargs={"pk": record.pk})
|
||||||
@ -144,7 +142,7 @@ class MembershipTable(tables.Table):
|
|||||||
# If the user has the right to manage the roles, display the link to manage them
|
# If the user has the right to manage the roles, display the link to manage them
|
||||||
roles = record.roles.all()
|
roles = record.roles.all()
|
||||||
s = ", ".join(str(role) for role in roles)
|
s = ", ".join(str(role) for role in roles)
|
||||||
if PermissionBackend.check_perm(get_current_request(), "member.change_membership_roles", record):
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "member.change_membership_roles", record):
|
||||||
s = format_html("<a href='" + str(reverse_lazy("member:club_manage_roles", kwargs={"pk": record.pk}))
|
s = format_html("<a href='" + str(reverse_lazy("member:club_manage_roles", kwargs={"pk": record.pk}))
|
||||||
+ "'>" + s + "</a>")
|
+ "'>" + s + "</a>")
|
||||||
return s
|
return s
|
||||||
@ -167,7 +165,7 @@ class ClubManagerTable(tables.Table):
|
|||||||
def render_user(self, value):
|
def render_user(self, value):
|
||||||
# If the user has the right, link the displayed user with the page of its detail.
|
# If the user has the right, link the displayed user with the page of its detail.
|
||||||
s = value.username
|
s = value.username
|
||||||
if PermissionBackend.check_perm(get_current_request(), "auth.view_user", value):
|
if PermissionBackend.check_perm(get_current_authenticated_user(), "auth.view_user", value):
|
||||||
s = format_html("<a href={url}>{name}</a>",
|
s = format_html("<a href={url}>{name}</a>",
|
||||||
url=reverse_lazy('member:user_detail', kwargs={"pk": value.pk}), name=s)
|
url=reverse_lazy('member:user_detail', kwargs={"pk": value.pk}), name=s)
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
{{ title }}
|
{{ title }}
|
||||||
</h3>
|
</h3>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<input id="searchbar" type="text" class="form-control" placeholder="Nom/prénom/note...">
|
<input id="searchbar" type="text" class="form-control" placeholder="Nom/prénom/note…">
|
||||||
<div class="form-check">
|
<div class="form-check">
|
||||||
<label class="form-check-label" for="only_active">
|
<label class="form-check-label" for="only_active">
|
||||||
<input type="checkbox" class="checkboxinput form-check-input" id="only_active"
|
<input type="checkbox" class="checkboxinput form-check-input" id="only_active"
|
||||||
|
@ -25,14 +25,6 @@
|
|||||||
</a>
|
</a>
|
||||||
</dd>
|
</dd>
|
||||||
|
|
||||||
<dt class="col-xl-6">{% trans 'friendships'|capfirst %}</dt>
|
|
||||||
<dd class="col-xl-6">
|
|
||||||
<a class="badge badge-secondary" href="{% url 'member:user_trust' user_object.pk %}">
|
|
||||||
<i class="fa fa-edit"></i>
|
|
||||||
{% trans 'Manage friendships' %} ({{ user_object.note.trusting.all|length }})
|
|
||||||
</a>
|
|
||||||
</dd>
|
|
||||||
|
|
||||||
{% if "member.view_profile"|has_perm:user_object.profile %}
|
{% if "member.view_profile"|has_perm:user_object.profile %}
|
||||||
<dt class="col-xl-6">{% trans 'section'|capfirst %}</dt>
|
<dt class="col-xl-6">{% trans 'section'|capfirst %}</dt>
|
||||||
<dd class="col-xl-6">{{ user_object.profile.section }}</dd>
|
<dd class="col-xl-6">{{ user_object.profile.section }}</dd>
|
||||||
@ -47,13 +39,13 @@
|
|||||||
<dt class="col-xl-6">{% trans 'address'|capfirst %}</dt>
|
<dt class="col-xl-6">{% trans 'address'|capfirst %}</dt>
|
||||||
<dd class="col-xl-6">{{ user_object.profile.address }}</dd>
|
<dd class="col-xl-6">{{ user_object.profile.address }}</dd>
|
||||||
|
|
||||||
<dt class="col-xl-6">{% trans 'paid'|capfirst %}</dt>
|
{% if user_object.note and "note.view_note"|has_perm:user_object.note %}
|
||||||
<dd class="col-xl-6">{{ user_object.profile.paid|yesno }}</dd>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if user_object.note and "note.view_note"|has_perm:user_object.note %}
|
|
||||||
<dt class="col-xl-6">{% trans 'balance'|capfirst %}</dt>
|
<dt class="col-xl-6">{% trans 'balance'|capfirst %}</dt>
|
||||||
<dd class="col-xl-6">{{ user_object.note.balance | pretty_money }}</dd>
|
<dd class="col-xl-6">{{ user_object.note.balance | pretty_money }}</dd>
|
||||||
|
|
||||||
|
<dt class="col-xl-6">{% trans 'paid'|capfirst %}</dt>
|
||||||
|
<dd class="col-xl-6">{{ user_object.profile.paid|yesno }}</dd>
|
||||||
|
{% endif %}
|
||||||
{% endif %}
|
{% endif %}
|
||||||
</dl>
|
</dl>
|
||||||
|
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user