from __future__ import annotations
from functools import cached_property
from sedate import utcnow
from sqlalchemy import func, desc
from onegov.core.collection import GenericCollection, Pagination
from onegov.fsi.models import (
CourseAttendee, CourseSubscription, CourseEvent, Course)
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from datetime import datetime
from onegov.fsi.request import FsiRequest
from sqlalchemy.orm import Query, Session
from typing import NamedTuple, TypeVar
from typing import Self
from uuid import UUID
class RankedSubscriptionRow(NamedTuple):
attendee_id: UUID
event_completed: bool
name: str
refresh_interval: int | None
course_id: UUID
start: datetime
end: datetime
rownum: int
class LastSubscriptionRow(NamedTuple):
attendee_id: UUID
start: datetime
end: datetime
name: str
event_completed: bool
refresh_interval: int | None
class AuditRow(NamedTuple):
id: UUID
first_name: str | None
last_name: str | None
organisation: str | None
source_id: str | None
start: datetime | None
end: datetime | None
event_completed: bool | None
refresh_interval: int | None
name: str | None
# FIXME: It's not quite kosher that we use a non-model row, so there
# may be some methods with incorrect types
[docs]
class AuditCollection(
GenericCollection['AuditRow'], # type:ignore[type-var]
Pagination['AuditRow'] # type:ignore[type-var]
):
"""
Displays the list of attendees filtered by a course and organisation
for evaluation if they subscribed and completed a course event.
The organisation filter should also be exact and not fuzzy.
"""
def __init__(
self,
session: Session,
course_id: UUID | None,
auth_attendee: CourseAttendee,
organisations: list[str] | None = None,
letter: str | None = None,
page: int = 0,
exclude_inactive: bool = True
) -> None:
super().__init__(session)
# When using the class link, the option with a course is still
[docs]
self.course_id = course_id if course_id else (
self.relevant_courses[0][0] if self.relevant_courses else None)
[docs]
self.auth_attendee = auth_attendee
# e.g. SD / STVA or nothing in case of editor
[docs]
self.organisations = organisations or []
[docs]
self.letter = letter.upper() if letter else None
[docs]
self.exclude_inactive = exclude_inactive
[docs]
def subset(self) -> Query[AuditRow]:
return self.query()
@property
[docs]
def page_index(self) -> int:
return self.page
[docs]
def page_by_index(self, index: int) -> Self:
return self.__class__(
self.session,
page=index,
course_id=self.course_id,
auth_attendee=self.auth_attendee,
organisations=self.organisations,
letter=self.letter,
exclude_inactive=self.exclude_inactive
)
[docs]
def by_letter_and_orgs(
self,
letter: str | None = None,
orgs: list[str] | None = None
) -> Self:
return self.__class__(
self.session,
page=0,
course_id=self.course_id,
auth_attendee=self.auth_attendee,
organisations=orgs if orgs is not None else self.organisations,
letter=letter or self.letter,
exclude_inactive=self.exclude_inactive
)
[docs]
def by_letter(self, letter: str | None) -> Self:
return self.__class__(
self.session,
page=0,
course_id=self.course_id,
auth_attendee=self.auth_attendee,
organisations=self.organisations,
exclude_inactive=self.exclude_inactive,
letter=letter,
)
[docs]
def __eq__(self, other: object) -> bool:
return (
isinstance(other, self.__class__)
and self.page == other.page
and self.course_id == other.course_id
and self.auth_attendee == other.auth_attendee
and self.organisations == other.organisations
and self.exclude_inactive == other.exclude_inactive
and self.letter == other.letter
)
[docs]
def ranked_subscription_query(
self,
past_only: bool = True
) -> Query[RankedSubscriptionRow]:
"""
Ranks all subscriptions of all events of a course
windowed over the attendee_id and ranked after completed, most recent
Use this query to make a join with any collection of attendees.
"""
ranked = self.session.query(
CourseSubscription.attendee_id,
CourseSubscription.event_completed,
Course.name,
Course.refresh_interval,
CourseEvent.course_id,
CourseEvent.start,
CourseEvent.end,
func.row_number().over(
partition_by=CourseSubscription.attendee_id,
order_by=[
desc(CourseSubscription.event_completed),
desc(CourseEvent.start)]
).label('rownum'),
)
ranked = ranked.join(
CourseEvent, CourseEvent.id == CourseSubscription.course_event_id)
ranked = ranked.join(
Course, Course.id == CourseEvent.course_id)
ranked = ranked.filter(
CourseEvent.course_id == self.course_id,
CourseSubscription.attendee_id.isnot(None)
)
if past_only:
ranked = ranked.filter(CourseEvent.start < utcnow())
return ranked
[docs]
def last_subscriptions(self) -> Query[LastSubscriptionRow]:
"""Retrieve the last completed subscription by attemdee for
a given the course_id.
"""
ranked = self.ranked_subscription_query().subquery('ranked')
subquery = self.session.query(
CourseSubscription.attendee_id,
ranked.c.start,
ranked.c.end,
ranked.c.name,
ranked.c.event_completed,
ranked.c.refresh_interval,
).select_entity_from(ranked)
return subquery.filter(ranked.c.rownum == 1)
[docs]
def filter_attendees_by_role(self, query: Query[T]) -> Query[T]:
"""Filter permissions of editor, exclude external, """
if self.auth_attendee.role == 'admin':
if not self.organisations:
return query
return query.filter(
CourseAttendee.organisation.in_(self.organisations)
)
else:
editors_permissions = self.auth_attendee.permissions or []
return query.filter(
CourseAttendee.organisation.in_(tuple(
p for p in editors_permissions
if p in self.organisations
) if self.organisations else editors_permissions)
)
[docs]
def query(self) -> Query[AuditRow]:
last = self.last_subscriptions().subquery()
query = self.session.query(
CourseAttendee.id,
CourseAttendee.first_name,
CourseAttendee.last_name,
CourseAttendee.organisation,
CourseAttendee.source_id,
last.c.start.label('start'),
last.c.end.label('end'),
last.c.event_completed,
last.c.refresh_interval,
last.c.name
)
if self.letter:
query = query.filter(
func.upper(
func.unaccent(CourseAttendee.last_name)
).startswith(self.letter)
)
query = self.filter_attendees_by_role(query)
if self.exclude_inactive:
query = query.filter(CourseAttendee.active == True)
query = query.join(
last, CourseAttendee.id == last.c.attendee_id, isouter=True)
query = query.order_by(
CourseAttendee.last_name,
CourseAttendee.first_name,
)
return query
[docs]
def next_subscriptions(
self,
request: FsiRequest
) -> dict[UUID, tuple[str, datetime]]:
next_subscriptions: dict[UUID, tuple[str, datetime]] = {}
if self.course_id:
# FIXME: We can do this in a single query, this is N+1...
query = request.session.query(CourseEvent)
query = query.filter(CourseEvent.course_id == self.course_id)
query = query.filter(CourseEvent.start >= utcnow())
query = query.order_by(CourseEvent.start)
for event in query:
attendees = event.attendees.with_entities(CourseAttendee.id)
for attendee in attendees:
next_subscriptions.setdefault(
attendee[0],
(request.link(event), event.start)
)
return next_subscriptions
@property
[docs]
def model_class(self) -> type[CourseAttendee]: # type:ignore[override]
return CourseAttendee
@cached_property
[docs]
def course(self) -> Course | None:
return self.session.query(Course).filter_by(
id=self.course_id).first() if self.course_id else None
@cached_property
[docs]
def used_letters(self) -> list[str]:
""" Returns a list of all the distinct first letters of the peoples
last names.
"""
letter = func.left(CourseAttendee.last_name, 1)
letter = func.upper(func.unaccent(letter))
query = self.session.query(letter.distinct().label('letter'))
query = query.order_by(letter)
return [r.letter for r in query if r.letter]
@cached_property
[docs]
def relevant_courses(self) -> tuple[tuple[UUID, str], ...]:
return tuple(self.session.query(Course.id, Course.name).filter(
Course.hidden_from_public == False,
Course.mandatory_refresh != None
))