Source code for feriennet.models.group_invite

from __future__ import annotations

from functools import cached_property
from onegov.activity.models import Attendee, Booking, Occasion, Period
from onegov.activity.utils import random_group_code
from onegov.user import User
from sqlalchemy import func, or_
from sqlalchemy.orm import joinedload


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Iterator
    from sqlalchemy.orm import Query, Session
    from typing import Self


[docs] class GroupInvite: def __init__( self, session: Session, group_code: str, username: str | None ) -> None:
[docs] self.session = session
[docs] self.group_code = group_code
[docs] self.username = username
@classmethod
[docs] def create(cls, session: Session, username: str | None) -> Self: """ Creates a new group invite with a code that is not yet used. """ candidate = cls( session=session, group_code=random_group_code(), username=username) # this might happen once in 26⁹ tries 🤞 if candidate.exists: return cls.create(session, username) return candidate
[docs] def for_username(self, username: str | None) -> Self: return self.__class__(self.session, self.group_code, username)
@cached_property
[docs] def user(self) -> User | None: if not self.username: return None return ( self.session.query(User) .filter_by(username=self.username).first() )
@property
[docs] def exists(self) -> bool: """ Returns True if the group_code associated with this invite exists. """ return self.session.query(self.bookings().exists()).scalar()
[docs] def bookings(self) -> Query[Booking]: """ Returns a query of the bookings associated with this invite. """ return ( self.session.query(Booking) .options(joinedload(Booking.attendee)) .options(joinedload(Booking.occasion)) .options(joinedload(Booking.period)) .filter_by(group_code=self.group_code) .filter(or_( Booking.state.in_(('open', 'accepted')), Period.confirmed == False )) )
@cached_property
[docs] def occasion(self) -> Occasion: """ Looks up the occasion linked to this group invite. Technically it would be possible that a group code points to multiple occasions, but that would be an error. If that happens, an exception will be thrown. """ return self.session.query(Occasion).filter(Occasion.id.in_( self.bookings().with_entities(Booking.occasion_id).subquery() )).one()
@cached_property
[docs] def attendees(self) -> tuple[tuple[Attendee, Booking], ...]: """ Returns the attendees linked to this invite. """ return tuple( (booking.attendee, booking) for booking in self.bookings() # FIXME: Why is this an outerjoin? attendee_id is not nullable # so a regular join should work just fine .outerjoin(Attendee) .order_by(func.unaccent(Attendee.name)) )
[docs] def prospects( self, username: str ) -> Iterator[tuple[Attendee, Booking | None]]: """ Returns the attendees associated with the given users that are not yet part of the group. The result is a list of tuples with the first element being the attendee and the second element being the booking for the linked occasion, if such a booking already exists (otherwise None). """ if not username: return existing = {a.id for a, b in self.attendees} attendees = ( self.session.query(Attendee) .filter(Attendee.username == username) .order_by(func.unaccent(Attendee.name)) ) bookings_query = ( self.session.query(Booking) .filter(Booking.occasion_id == self.occasion.id) .filter(Booking.attendee_id.in_( attendees.with_entities(Attendee.id).subquery())) ) bookings = {b.attendee_id: b for b in bookings_query} for attendee in attendees: if attendee.id not in existing: yield attendee, bookings.get(attendee.id, None)