Source code for fsi.forms.course_attendee
from __future__ import annotations
from functools import cached_property
from onegov.form import Form
from onegov.form.fields import ChosenSelectMultipleField, ChosenSelectField
from onegov.fsi import _
from onegov.fsi.models import CourseAttendee
from onegov.fsi.models.course_attendee import external_attendee_org
from onegov.user import User
from wtforms.fields import EmailField
from wtforms.fields import StringField
from wtforms.validators import InputRequired, Email
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from onegov.fsi.request import FsiRequest
[docs]
class CourseAttendeeForm(Form):
@cached_property
[docs]
def unique_permission_codes(self) -> tuple[str, ...]:
return tuple(
code for code, in self.request.session.query(
CourseAttendee.organisation.distinct().label('code')
).order_by(CourseAttendee.organisation).filter(
CourseAttendee.organisation.isnot(None)
)
)
[docs]
def update_model(self, model: CourseAttendee) -> None:
model.first_name = self.first_name.data
model.last_name = self.last_name.data
if self.permissions:
model.permissions = self.permissions.data
if self.email:
model._email = self.email.data
if self.organisation:
model.organisation = self.organisation.data
[docs]
def apply_model(self, model: CourseAttendee) -> None:
self.first_name.data = model.first_name
self.last_name.data = model.last_name
if self.permissions:
self.permissions.data = model.permissions
if self.email:
self.email.data = model.email
if self.organisation:
self.organisation.data = model.organisation
[docs]
def set_organisation_choices(self) -> None:
if self.request.is_admin:
organisations = self.unique_permission_codes
else:
assert self.request.attendee is not None
organisations = tuple(self.request.attendee.permissions or ())
if external_attendee_org not in organisations:
organisations = (external_attendee_org, *organisations)
self.organisation.choices = [(org, org) for org in organisations]
[docs]
def on_request(self) -> None:
# is an external
if not self.model.user_id or self.model.user.role != 'editor':
self.delete_field('permissions')
else:
self.delete_field('email')
self.permissions.choices = [
(code, code) for code in self.unique_permission_codes
]
if self.model.user_id or not self.request.is_manager:
self.delete_field('organisation')
else:
self.set_organisation_choices()
[docs]
class AddExternalAttendeeForm(CourseAttendeeForm):
[docs]
def ensure_email_not_existing(self) -> bool:
email = self.email.data
session = self.request.session
# NOTE: Even though we potentially perform a redundant EXISTS query
# it's still almost certainly faster, since we only have to
# submit a single query to the database, plus the database is
# probably smart enough to optimize the second query away if
# the first one returned TRUE and if it doesn't do that it's
# probably because it's faster this way (parallelism).
att = session.query(CourseAttendee).filter_by(_email=email)
user = session.query(User).filter_by(username=email)
if session.query(att.exists() | user.exists()).scalar():
assert isinstance(self.email.errors, list)
self.email.errors.append(
_('An attendee with this email already exists'))
return False
return True
[docs]
def on_request(self) -> None:
self.delete_field('permissions')
self.set_organisation_choices()