8
0
Fork 0
mirror of https://gitlab2.federez.net/re2o/re2o synced 2024-11-30 00:22:25 +00:00
re2o/users/forms.py

1087 lines
36 KiB
Python
Raw Normal View History

# -*- mode: python; coding: utf-8 -*-
2020-11-23 16:06:37 +00:00
# Re2o est un logiciel d'administration développé initiallement au Rézo Metz. Il
2017-01-15 23:01:18 +00:00
# se veut agnostique au réseau considéré, de manière à être installable en
# quelques clics.
#
2020-04-17 14:48:27 +00:00
# Copyright © 2017-2020 Gabriel Détraz
# Copyright © 2017-2020 Lara Kermarec
# Copyright © 2017-2020 Augustin Lemesle
# Copyright © 2017-2020 Hugo Levy--Falk
# Copyright © 2017-2020 Jean-Romain Garnier
2017-01-15 23:01:18 +00:00
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
2017-10-14 18:18:12 +00:00
"""
2020-05-29 22:33:29 +00:00
Forms for the 'users' app of re2o. It highly depends on
:users:models and is mainly used by :users:views.
2017-01-15 23:01:18 +00:00
2020-05-29 22:33:29 +00:00
The following forms are mainly used to create, edit or delete
anything related to 'users' :
* Adherent (personnal data)
* Club
* Ban
* ServiceUser
* Whitelists
* ...
See the details for each of these operations in the documentation
of each of the method.
2017-10-14 18:18:12 +00:00
"""
from __future__ import unicode_literals
2020-11-17 12:54:33 +00:00
from os import walk, path
from django import forms
from django.forms import ModelForm, Form
from django.contrib.auth.forms import ReadOnlyPasswordHashField
2020-12-31 19:12:36 +00:00
from django.contrib.auth.password_validation import (
validate_password,
password_validators_help_text_html,
)
2016-11-21 19:14:25 +00:00
from django.core.validators import MinLengthValidator
2020-11-17 12:54:33 +00:00
from django.conf import settings
from django.utils import timezone
2019-09-05 17:55:54 +00:00
from django.utils.functional import lazy
2017-12-31 16:11:19 +00:00
from django.contrib.auth.models import Group, Permission
2018-08-15 17:15:26 +00:00
from django.utils.translation import ugettext_lazy as _
2018-09-20 13:38:36 +00:00
from django.utils.safestring import mark_safe
from machines.models import Interface, Machine, Nas
from topologie.models import Port
2017-10-14 18:18:12 +00:00
from preferences.models import OptionalUser
from re2o.utils import remove_user_room
from re2o.base import get_input_formats_help_text
from re2o.mixins import FormRevMixin
from re2o.widgets import (
AutocompleteMultipleModelWidget,
AutocompleteModelWidget,
2020-12-31 19:12:36 +00:00
)
2018-04-14 23:16:49 +00:00
from re2o.field_permissions import FieldPermissionFormMixin
2018-09-20 08:57:31 +00:00
from preferences.models import GeneralOption
2018-05-05 15:58:13 +00:00
from .widgets import DateTimePicker
from .models import (
User,
ServiceUser,
School,
ListRight,
Whitelist,
2018-08-01 11:06:25 +00:00
EMailAddress,
ListShell,
Ban,
Adherent,
Club,
)
#### Django Admin Custom Views
2020-04-16 20:06:14 +00:00
2020-05-28 20:08:16 +00:00
class UserAdminForm(FormRevMixin, forms.ModelForm):
2020-05-28 20:24:07 +00:00
"""A form for creating new and editing users. Includes all the required
2017-10-14 18:18:12 +00:00
fields, plus a repeated password.
Parameters:
DjangoForm : Inherit from basic django form
"""
2017-10-14 18:18:12 +00:00
password1 = forms.CharField(
2018-08-15 17:15:26 +00:00
label=_("Password"),
2017-10-14 18:18:12 +00:00
widget=forms.PasswordInput,
max_length=255,
2020-05-28 20:08:16 +00:00
help_text=password_validators_help_text_html(),
required=False,
2017-10-14 18:18:12 +00:00
)
password2 = forms.CharField(
2018-08-15 17:15:26 +00:00
label=_("Password confirmation"),
2017-10-14 18:18:12 +00:00
widget=forms.PasswordInput,
max_length=255,
2020-05-28 20:08:16 +00:00
required=False,
2017-10-14 18:18:12 +00:00
)
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
2020-05-28 20:08:16 +00:00
super(UserAdminForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["email"].required = True
class Meta:
2020-05-28 20:08:16 +00:00
fields = ("pseudo", "surname", "name", "email", "is_superuser")
def clean_password2(self):
"""Clean password 2, check if passwd1 and 2 values match.
Parameters:
self : Apply on a django Form UserCreationForm instance
Returns:
password2 (string): The password2 value if all tests returned True
"""
password1 = self.cleaned_data.get("password1")
password2 = self.cleaned_data.get("password2")
2020-05-28 20:08:16 +00:00
if password1 and password2:
if password1 and password2 and password1 != password2:
raise forms.ValidationError(_("The passwords don't match."))
validate_password(password1)
return password2
def save(self, commit=True):
"""Save function. Call standard "set_password" django function,
from provided value for new password, for making hash.
Parameters:
self : Apply on a django Form UserCreationForm instance
commit : If False, don't make the real save in database
"""
# Save the provided password in hashed format
2020-05-28 20:08:16 +00:00
user = super(UserAdminForm, self).save(commit=False)
if self.cleaned_data["password1"]:
user.set_password(self.cleaned_data["password1"])
user.save()
return user
2017-10-14 18:18:12 +00:00
2020-05-28 20:24:07 +00:00
class ServiceUserAdminForm(FormRevMixin, forms.ModelForm):
"""A form for creating new service users. Includes all the required
fields, plus a repeated password. For Admin view purpose only.
2017-10-14 18:18:12 +00:00
Parameters:
DjangoForm : Inherit from basic django form
"""
2017-10-14 18:18:12 +00:00
password1 = forms.CharField(
2020-12-31 19:12:36 +00:00
label=_("Password"), widget=forms.PasswordInput, max_length=255
2017-10-14 18:18:12 +00:00
)
password2 = forms.CharField(
2020-12-31 19:12:36 +00:00
label=_("Password confirmation"), widget=forms.PasswordInput, max_length=255
2017-10-14 18:18:12 +00:00
)
2016-07-31 03:03:07 +00:00
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
2020-05-28 20:24:07 +00:00
super(ServiceUserAdminForm, self).__init__(*args, prefix=prefix, **kwargs)
2016-07-31 03:03:07 +00:00
class Meta:
model = ServiceUser
fields = ("pseudo",)
2016-07-31 03:03:07 +00:00
def clean_password2(self):
"""Clean password 2, check if passwd1 and 2 values match.
Parameters:
self : Apply on a django Form UserCreationForm instance
Returns:
password2 (string): The password2 value if all tests returned True
"""
2016-07-31 03:03:07 +00:00
password1 = self.cleaned_data.get("password1")
password2 = self.cleaned_data.get("password2")
if password1 and password2 and password1 != password2:
2018-08-15 17:15:26 +00:00
raise forms.ValidationError(_("The passwords don't match."))
2016-07-31 03:03:07 +00:00
return password2
def save(self, commit=True):
"""Save function. Call standard "set_password" django function,
from provided value for new password, for making hash.
Parameters:
self : Apply on a django Form ServiceUserAdminForm instance
commit : If False, don't make the real save in database
"""
2020-05-28 20:24:07 +00:00
user = super(ServiceUserAdminForm, self).save(commit=False)
2016-07-31 03:03:07 +00:00
user.set_password(self.cleaned_data["password1"])
user.save()
return user
2017-10-14 18:18:12 +00:00
### Classic Django View
class PassForm(FormRevMixin, FieldPermissionFormMixin, forms.ModelForm):
"""Django form for changing password, check if 2 passwords are the same,
and validate password for django base password validators provided in
settings_local.
Parameters:
DjangoForm : Inherit from basic django form
"""
2020-12-31 19:12:36 +00:00
selfpasswd = forms.CharField(
label=_("Current password"), max_length=255, widget=forms.PasswordInput
)
passwd1 = forms.CharField(
label=_("New password"),
max_length=255,
widget=forms.PasswordInput,
2020-12-31 19:12:36 +00:00
help_text=password_validators_help_text_html(),
)
passwd2 = forms.CharField(
2020-12-31 19:12:36 +00:00
label=_("New password confirmation"), max_length=255, widget=forms.PasswordInput
)
class Meta:
model = User
fields = []
def clean_passwd2(self):
"""Clean password 2, check if passwd1 and 2 values match, and
apply django validator with validate_password function.
Parameters:
self : Apply on a django Form PassForm instance
Returns:
password2 (string): The password2 value if all tests returned True
"""
password1 = self.cleaned_data.get("passwd1")
password2 = self.cleaned_data.get("passwd2")
if password1 and password2 and password1 != password2:
raise forms.ValidationError(_("The new passwords don't match."))
validate_password(password1, user=self.instance)
return password2
def clean_selfpasswd(self):
"""Clean selfpassword, check if provided original user password match
with the stored value.
Parameters:
self : Apply on a django Form PassForm instance
"""
if not self.instance.check_password(self.cleaned_data.get("selfpasswd")):
raise forms.ValidationError(_("The current password is incorrect."))
return
def save(self, commit=True):
"""Save function. Call standard "set_password" django function,
and call set_active for set user in active state if needed.
Parameters:
self : Apply on a django Form PassForm instance
commit : If False, don't make the real save in database
"""
user = super(PassForm, self).save(commit=False)
user.set_password(self.cleaned_data.get("passwd1"))
user.save()
2016-07-20 10:06:33 +00:00
class ResetPasswordForm(forms.Form):
"""A form for asking to reset password.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2018-08-15 17:15:26 +00:00
pseudo = forms.CharField(label=_("Username"), max_length=255)
2016-07-20 10:06:33 +00:00
email = forms.EmailField(max_length=255)
2017-10-14 18:18:12 +00:00
class MassArchiveForm(forms.Form):
"""A form for archiving a lot de users. Get a start date
for start archiving.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
date = forms.DateTimeField(help_text="%d/%m/%y")
full_archive = forms.BooleanField(
2020-12-31 19:12:36 +00:00
label=_("Fully archive users? WARNING: CRITICAL OPERATION IF TRUE"),
initial=False,
required=False,
)
def clean(self):
2017-10-14 18:18:12 +00:00
cleaned_data = super(MassArchiveForm, self).clean()
date = cleaned_data.get("date")
if date:
if date > timezone.now():
raise forms.ValidationError(
_(
"Impossible to archive users"
" whose end access date is in"
" the future."
)
)
2017-10-14 18:18:12 +00:00
class AdherentForm(FormRevMixin, FieldPermissionFormMixin, ModelForm):
"""Adherent Edition Form, base form used for editing user by himself
or another user. Labels are provided for help purposes.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(AdherentForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["name"].label = _("First name")
self.fields["surname"].label = _("Surname")
self.fields["email"].label = _("Email address")
self.fields["school"].label = _("School")
self.fields["comment"].label = _("Comment")
if "room" in self.fields:
self.fields["room"].label = _("Room")
self.fields["room"].empty_label = _("No room")
self.fields["school"].empty_label = _("Select a school")
2018-07-30 15:00:41 +00:00
class Meta:
model = Adherent
fields = [
"name",
"surname",
"pseudo",
"email",
"school",
"comment",
"telephone",
"room",
]
widgets = {
"school": AutocompleteModelWidget(url="/users/school-autocomplete"),
"room": AutocompleteModelWidget(
url="/topologie/room-autocomplete",
2020-12-31 19:12:36 +00:00
attrs={
"data-minimum-input-length": 3 # Only trigger autocompletion after 3 characters have been typed
},
),
"shell": AutocompleteModelWidget(url="/users/shell-autocomplete"),
}
force = forms.BooleanField(
label=_("Force the move?"), initial=False, required=False
)
def clean_telephone(self):
"""Clean telephone, check if telephone is made mandatory, and
raise error if not provided
Parameters:
self : Apply on a django Form AdherentForm instance
Returns:
telephone (string): The telephone string if clean is True
"""
telephone = self.cleaned_data["telephone"]
if not telephone and OptionalUser.get_cached_value("is_tel_mandatory"):
raise forms.ValidationError(_("A valid telephone number is required."))
return telephone
def clean_force(self):
"""Clean force, remove previous user from room if needed.
Parameters:
self : Apply on a django Form AdherentForm instance
"""
room = self.cleaned_data.get("room")
if self.cleaned_data.get("force", False) and room:
2019-09-10 15:09:26 +00:00
remove_user_room(room)
return
2017-10-14 18:18:12 +00:00
2020-04-15 19:04:14 +00:00
def clean_room(self):
"""Clean room, based on room policy provided by preferences.
If needed, call remove_user_room to make the room empty before
saving self.instance into that room.
Parameters:
self : Apply on a django Form AdherentForm instance
Returns:
room (string): The room instance
"""
2020-04-15 19:04:14 +00:00
# Handle case where regular users can force move
room = self.cleaned_data.get("room")
room_policy = OptionalUser.get_cached_value("self_room_policy")
if room_policy == OptionalUser.DISABLED or not room:
2020-04-15 19:04:14 +00:00
return room
# Remove the previous user's room, if allowed and necessary
remove_user_room(room, force=bool(room_policy == OptionalUser.ALL_ROOM))
2020-04-15 19:04:14 +00:00
# Run standard clean process
return room
class AdherentCreationForm(AdherentForm):
"""AdherentCreationForm. Inherit from AdherentForm, base form used for creating
user by himself or another user. Labels are provided for help purposes.
Add some instructions, and validation for initial creation.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2020-04-17 16:01:35 +00:00
# Champ pour choisir si un lien est envoyé par mail pour le mot de passe
init_password_by_mail_info = _(
"If this options is set, you will receive a link to set"
" your initial password by email. If you do not have"
" any means of accessing your emails, you can disable"
" this option to set your password immediatly."
" You will still receive an email to confirm your address."
" Failure to confirm your address will result in an"
" automatic suspension of your account until you do."
)
2020-04-17 16:01:35 +00:00
init_password_by_mail = forms.BooleanField(
2020-12-31 19:12:36 +00:00
help_text=init_password_by_mail_info, required=False, initial=True
2020-04-17 16:01:35 +00:00
)
init_password_by_mail.label = _("Send password reset link by email.")
# Champs pour initialiser le mot de passe
# Validators are handled manually since theses fields aren't always required
password1 = forms.CharField(
required=False,
label=_("Password"),
widget=forms.PasswordInput,
max_length=255,
2020-12-31 19:12:36 +00:00
help_text=password_validators_help_text_html(),
2020-04-17 16:01:35 +00:00
)
password2 = forms.CharField(
required=False,
label=_("Password confirmation"),
widget=forms.PasswordInput,
max_length=255,
)
2018-09-19 21:24:15 +00:00
# Champ permettant d'éviter au maxium les doublons d'utilisateurs
former_user_check_info = _(
2019-11-20 00:52:11 +00:00
"If you already have an account, please use it. If your lost access to"
" it, please consider using the forgotten password button on the"
" login page or contacting support."
)
former_user_check = forms.BooleanField(
required=True, help_text=former_user_check_info
)
2019-11-20 00:52:11 +00:00
former_user_check.label = _("I certify that I have not had an account before.")
2018-09-20 08:57:31 +00:00
# Checkbox for GTU
gtu_check = forms.BooleanField(required=True)
class Meta(AdherentForm.Meta):
model = Adherent
fields = [
"name",
"surname",
"pseudo",
"email",
"school",
"comment",
"telephone",
"room",
"state",
]
2018-09-19 21:24:15 +00:00
def __init__(self, *args, **kwargs):
super(AdherentCreationForm, self).__init__(*args, **kwargs)
gtu_file = GeneralOption.get_cached_value("GTU")
self.fields["email"].required = True
self.fields["gtu_check"].label = mark_safe(
"%s <a href='%s' download='CGU'>%s</a>."
% (
2019-09-05 17:55:54 +00:00
_("I commit to accept the"),
2019-09-20 12:20:42 +00:00
gtu_file.url if gtu_file else "#",
_("General Terms of Use"),
2019-09-05 17:55:54 +00:00
)
)
2020-04-17 16:01:35 +00:00
# Remove password fields if option is disabled
if not OptionalUser.get_cached_value("allow_set_password_during_user_creation"):
self.fields.pop("init_password_by_mail")
self.fields.pop("password1")
self.fields.pop("password2")
def clean_password2(self):
"""Clean password 2, check if passwd1 and 2 values match, and
apply django validator with validate_password function.
Parameters:
self : Apply on a django Form AdherentCreationForm instance
Returns:
password2 (string): The password2 value if all tests returned True
"""
send_email = self.cleaned_data.get("init_password_by_mail")
if send_email:
return None
# Check that the two password entries match
password1 = self.cleaned_data.get("password1")
password2 = self.cleaned_data.get("password2")
if password1 and password2 and password1 != password2:
raise forms.ValidationError(_("The passwords don't match."))
validate_password(password1)
return password2
def save(self, commit=True):
"""Save function. If password has been set during creation,
call standard "set_password" django function from provided value
for new password, for making hash.
Parameters:
self : Apply on a django Form AdherentCreationForm instance
commit : If False, don't make the real save in database
"""
# Save the provided password in hashed format
user = super(AdherentForm, self).save(commit=False)
2020-12-31 19:12:36 +00:00
is_set_password_allowed = OptionalUser.get_cached_value(
"allow_set_password_during_user_creation"
)
set_passwd = is_set_password_allowed and not self.cleaned_data.get(
"init_password_by_mail"
)
2020-04-17 18:41:02 +00:00
if set_passwd:
user.set_password(self.cleaned_data["password1"])
user.save()
return user
2019-09-05 17:55:54 +00:00
class AdherentEditForm(AdherentForm):
"""AdherentEditForm. Inherit from AdherentForm, base form used for editing
user by himself or another user. Labels are provided for help purposes.
Add some instructions, and validation, fields depends on editing user rights.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
super(AdherentEditForm, self).__init__(*args, **kwargs)
self.fields["gpg_fingerprint"].widget.attrs["placeholder"] = _(
"Leave empty if you don't have any GPG key."
)
2020-04-21 14:47:09 +00:00
self.user = kwargs["instance"]
2020-04-21 16:19:03 +00:00
self.fields["email"].required = bool(self.user.email)
if "shell" in self.fields:
self.fields["shell"].empty_label = _("Default shell")
class Meta(AdherentForm.Meta):
model = Adherent
fields = [
"name",
"surname",
"pseudo",
"email",
"school",
"comment",
"telephone",
"room",
"shell",
"gpg_fingerprint",
"shortcuts_enabled",
]
class ClubForm(FormRevMixin, FieldPermissionFormMixin, ModelForm):
"""ClubForm. For editing club by himself or another user. Labels are provided for
help purposes. Add some instructions, and validation, fields depends
on editing user rights.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(ClubForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["surname"].label = _("Name")
self.fields["school"].label = _("School")
self.fields["comment"].label = _("Comment")
self.fields["email"].label = _("Email address")
if "room" in self.fields:
self.fields["room"].label = _("Room")
self.fields["room"].empty_label = _("No room")
self.fields["school"].empty_label = _("Select a school")
self.fields["mailing"].label = _("Use a mailing list")
class Meta:
model = Club
fields = [
"surname",
"pseudo",
"school",
"comment",
"room",
"email",
"telephone",
"email",
"shell",
"mailing",
]
widgets = {
"school": AutocompleteModelWidget(url="/users/school-autocomplete"),
"room": AutocompleteModelWidget(url="/topologie/room-autocomplete"),
"shell": AutocompleteModelWidget(url="/users/shell-autocomplete"),
}
def clean_telephone(self):
"""Clean telephone, check if telephone is made mandatory, and
raise error if not provided
Parameters:
self : Apply on a django Form ClubForm instance
Returns:
telephone (string): The telephone string if clean is True
"""
telephone = self.cleaned_data["telephone"]
if not telephone and OptionalUser.get_cached_value("is_tel_mandatory"):
raise forms.ValidationError(_("A valid telephone number is required."))
return telephone
class ClubAdminandMembersForm(FormRevMixin, ModelForm):
"""ClubAdminandMembersForm. Only For editing administrators of a club by himself
or another user.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta:
model = Club
fields = ["administrators", "members"]
2020-12-28 20:06:24 +00:00
widgets = {
"administrators": AutocompleteMultipleModelWidget(
2020-12-31 19:12:36 +00:00
url="/users/adherent-autocomplete"
2020-12-28 20:06:24 +00:00
),
"members": AutocompleteMultipleModelWidget(
2020-12-31 19:12:36 +00:00
url="/users/adherent-autocomplete"
),
2020-12-28 20:06:24 +00:00
}
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(ClubAdminandMembersForm, self).__init__(*args, prefix=prefix, **kwargs)
class PasswordForm(FormRevMixin, ModelForm):
"""PasswordForm. Do not use directly in views without extra validations.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta:
model = User
fields = ["password", "pwd_ntlm"]
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(PasswordForm, self).__init__(*args, prefix=prefix, **kwargs)
2017-10-14 18:18:12 +00:00
class ServiceUserForm(FormRevMixin, ModelForm):
"""ServiceUserForm, used for creating a service user, require
a password and set it.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2017-10-14 18:18:12 +00:00
password = forms.CharField(
2018-08-15 17:15:26 +00:00
label=_("New password"),
2017-10-14 18:18:12 +00:00
max_length=255,
validators=[MinLengthValidator(8)],
widget=forms.PasswordInput,
required=True,
2017-10-14 18:18:12 +00:00
)
class Meta:
model = ServiceUser
fields = ("pseudo", "access_group", "comment")
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(ServiceUserForm, self).__init__(*args, prefix=prefix, **kwargs)
def save(self, commit=True):
"""Save function. If password has been changed and provided,
call standard "set_password" django function from provided value
for new password, for making hash.
Parameters:
self : Apply on a django Form ServiceUserForm instance
commit : If False, don't make the real save in database
"""
user = super(ServiceUserForm, self).save(commit=False)
if self.cleaned_data["password"]:
user.set_password(self.cleaned_data.get("password"))
user.save()
2017-10-14 18:18:12 +00:00
class EditServiceUserForm(ServiceUserForm):
"""EditServiceUserForm, used for editing a service user, can
edit password, access_group and comment.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2018-12-22 12:10:31 +00:00
password = forms.CharField(
label=_("New password"),
max_length=255,
validators=[MinLengthValidator(8)],
widget=forms.PasswordInput,
required=False,
2018-12-22 12:10:31 +00:00
)
class Meta(ServiceUserForm.Meta):
fields = ["access_group", "comment"]
2017-10-14 18:18:12 +00:00
class StateForm(FormRevMixin, ModelForm):
"""StateForm, Change state of an user, and if
its main email is verified or not
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta:
model = User
2020-04-17 22:29:50 +00:00
fields = ["state", "email_state"]
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(StateForm, self).__init__(*args, prefix=prefix, **kwargs)
2020-04-17 22:29:50 +00:00
self.fields["state"].label = _("State")
self.fields["email_state"].label = _("Email state")
2018-05-03 12:22:52 +00:00
class GroupForm(FieldPermissionFormMixin, FormRevMixin, ModelForm):
"""GroupForm, form used for editing user groups.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2017-12-31 16:11:19 +00:00
groups = forms.ModelMultipleChoiceField(
Group.objects.all(), widget=forms.CheckboxSelectMultiple, required=False
2017-12-31 16:11:19 +00:00
)
class Meta:
model = User
fields = ["is_superuser", "groups"]
2017-12-31 16:11:19 +00:00
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
2017-12-31 16:11:19 +00:00
super(GroupForm, self).__init__(*args, prefix=prefix, **kwargs)
if "is_superuser" in self.fields:
self.fields["is_superuser"].label = _("Superuser")
2017-12-31 16:11:19 +00:00
class SchoolForm(FormRevMixin, ModelForm):
"""SchoolForm, form used for creating or editing school.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta:
model = School
fields = ["name"]
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(SchoolForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["name"].label = _("School")
2017-10-14 18:18:12 +00:00
class ShellForm(FormRevMixin, ModelForm):
"""ShellForm, form used for creating or editing shell.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta:
model = ListShell
fields = ["shell"]
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(ShellForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["shell"].label = _("Shell name")
class ListRightForm(FormRevMixin, ModelForm):
"""ListRightForm, form used for editing a listright,
related with django group object. Gid, primary key, can't
be edited.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2017-12-31 16:11:19 +00:00
permissions = forms.ModelMultipleChoiceField(
Permission.objects.all().select_related("content_type"),
2017-12-31 16:11:19 +00:00
widget=forms.CheckboxSelectMultiple,
required=False,
2017-12-31 16:11:19 +00:00
)
class Meta:
model = ListRight
fields = ("name", "unix_name", "critical", "permissions", "details")
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(ListRightForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["unix_name"].label = _("Name of the group of rights")
2017-10-14 18:18:12 +00:00
class NewListRightForm(ListRightForm):
"""ListRightForm, form used for creating a listright,
related with django group object.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
class Meta(ListRightForm.Meta):
fields = ("name", "unix_name", "gid", "critical", "permissions", "details")
def __init__(self, *args, **kwargs):
super(NewListRightForm, self).__init__(*args, **kwargs)
self.fields["gid"].label = _(
2019-11-20 00:52:11 +00:00
"GID. Warning: this field must not be edited after creation."
)
2017-10-14 18:18:12 +00:00
class DelListRightForm(Form):
"""DelListRightForm, form for deleting one or several ListRight
instances.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2017-10-14 18:18:12 +00:00
listrights = forms.ModelMultipleChoiceField(
2017-12-13 17:35:16 +00:00
queryset=ListRight.objects.none(),
2018-08-15 17:15:26 +00:00
label=_("Current groups of rights"),
widget=forms.CheckboxSelectMultiple,
2017-10-14 18:18:12 +00:00
)
2017-12-13 17:35:16 +00:00
def __init__(self, *args, **kwargs):
instances = kwargs.pop("instances", None)
2017-12-13 17:35:16 +00:00
super(DelListRightForm, self).__init__(*args, **kwargs)
if instances:
self.fields["listrights"].queryset = instances
2017-12-13 17:35:16 +00:00
else:
self.fields["listrights"].queryset = ListRight.objects.all()
2017-12-13 17:35:16 +00:00
class DelSchoolForm(Form):
"""DelSchoolForm, form for deleting one or several School
instances.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
2017-10-14 18:18:12 +00:00
schools = forms.ModelMultipleChoiceField(
2017-12-13 17:35:16 +00:00
queryset=School.objects.none(),
2018-08-15 17:15:26 +00:00
label=_("Current schools"),
widget=forms.CheckboxSelectMultiple,
2017-10-14 18:18:12 +00:00
)
2017-12-12 03:33:50 +00:00
def __init__(self, *args, **kwargs):
instances = kwargs.pop("instances", None)
2017-12-12 03:33:50 +00:00
super(DelSchoolForm, self).__init__(*args, **kwargs)
2017-12-13 17:35:16 +00:00
if instances:
self.fields["schools"].queryset = instances
2017-12-13 17:35:16 +00:00
else:
self.fields["schools"].queryset = School.objects.all()
2017-12-12 03:33:50 +00:00
class BanForm(FormRevMixin, ModelForm):
"""BanForm, form used for creating or editing a ban instance.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(BanForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["date_end"].label = _("End date")
self.fields["date_end"].localize = False
class Meta:
model = Ban
exclude = ["user"]
widgets = {"date_end": DateTimePicker}
class WhitelistForm(FormRevMixin, ModelForm):
"""WhitelistForm, form used for creating or editing a whitelist instance.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
super(WhitelistForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["date_end"].label = _("End date")
self.fields["date_end"].localize = False
class Meta:
model = Whitelist
exclude = ["user"]
widgets = {"date_end": DateTimePicker}
2018-08-01 11:06:25 +00:00
class EMailAddressForm(FormRevMixin, ModelForm):
"""EMailAddressForm, form used for creating or editing a local
email for a user.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
2018-08-01 11:06:25 +00:00
super(EMailAddressForm, self).__init__(*args, prefix=prefix, **kwargs)
self.fields["local_part"].label = _("Local part of the email address")
2019-11-20 00:52:11 +00:00
self.fields["local_part"].help_text = _("Can't contain @.")
def clean_local_part(self):
return self.cleaned_data.get("local_part").lower()
class Meta:
2018-08-01 11:06:25 +00:00
model = EMailAddress
exclude = ["user"]
2018-07-30 15:00:41 +00:00
class EmailSettingsForm(FormRevMixin, FieldPermissionFormMixin, ModelForm):
"""EMailSettingsForm, form used for editing email settings for a user.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
def __init__(self, *args, **kwargs):
prefix = kwargs.pop("prefix", self.Meta.model.__name__)
2018-07-30 15:00:41 +00:00
super(EmailSettingsForm, self).__init__(*args, prefix=prefix, **kwargs)
2020-04-21 14:47:09 +00:00
self.user = kwargs["instance"]
self.fields["email"].label = _("Main email address")
2020-04-21 16:19:03 +00:00
self.fields["email"].required = bool(self.user.email)
if "local_email_redirect" in self.fields:
self.fields["local_email_redirect"].label = _("Redirect local emails")
if "local_email_enabled" in self.fields:
self.fields["local_email_enabled"].label = _("Use local emails")
class Meta:
2018-06-29 14:36:04 +00:00
model = User
fields = ["email", "local_email_enabled", "local_email_redirect"]
2018-08-15 17:15:26 +00:00
class InitialRegisterForm(forms.Form):
"""InitialRegisterForm, form used for auto-register of room and mac-address
with captive-portal.
Parameters:
DjangoForm : Inherit from basic django form
2020-12-31 19:12:36 +00:00
"""
register_room = forms.BooleanField(required=False)
register_machine = forms.BooleanField(required=False)
def __init__(self, *args, **kwargs):
switch_ip = kwargs.pop("switch_ip")
switch_port = kwargs.pop("switch_port")
client_mac = kwargs.pop("client_mac")
self.user = kwargs.pop("user")
if switch_ip and switch_port:
# Looking for a port
port = Port.objects.filter(
switch__interface__ipv4__ipv4=switch_ip, port=switch_port
).first()
# If a port exists, checking there is a room AND radius
if port:
if (
port.get_port_profile.radius_type != "NO"
and port.get_port_profile.radius_mode == "STRICT"
and hasattr(port, "room")
):
# Requesting user is not in this room ?
if self.user.room != port.room:
self.new_room = port.room
if client_mac and switch_ip:
# If this interface doesn't already exists
if not Interface.objects.filter(mac_address=client_mac):
self.mac_address = client_mac
self.nas_type = Nas.objects.filter(
nas_type__interface__ipv4__ipv4=switch_ip
).first()
super(InitialRegisterForm, self).__init__(*args, **kwargs)
if hasattr(self, "new_room"):
self.fields["register_room"].label = _("This room is my room")
else:
self.fields.pop("register_room")
if hasattr(self, "mac_address"):
self.fields["register_machine"].label = _(
"This new connected device is mine"
)
else:
self.fields.pop("register_machine")
def clean_register_room(self):
"""Clean room, call remove_user_room to make the room empty before
saving self.instance into that room.
Parameters:
self : Apply on a django Form InitialRegisterForm instance
"""
if self.cleaned_data["register_room"]:
if self.user.is_class_adherent:
remove_user_room(self.new_room)
user = self.user.adherent
user.room = self.new_room
user.save()
if self.user.is_class_club:
user = self.user.club
user.room = self.new_room
user.save()
def clean_register_machine(self):
"""Clean register room, autoregister machine from user request mac_address.
Parameters:
self : Apply on a django Form InitialRegisterForm instance
"""
if self.cleaned_data["register_machine"]:
if self.mac_address and self.nas_type:
self.user.autoregister_machine(self.mac_address, self.nas_type)
2020-11-17 12:54:33 +00:00
class ThemeForm(FormRevMixin, forms.Form):
"""Form to change the theme of a user.
"""
theme = forms.ChoiceField(widget=forms.Select())
def __init__(self, *args, **kwargs):
2020-12-31 19:12:36 +00:00
_, _, themes = next(walk(path.join(settings.STATIC_ROOT, "css/themes")))
2020-11-17 12:54:33 +00:00
if not themes:
themes = ["default.css"]
super(ThemeForm, self).__init__(*args, **kwargs)
2020-12-31 19:12:36 +00:00
self.fields["theme"].choices = [(theme, theme) for theme in themes]