Source code for feriennet.exports.unlucky

from __future__ import annotations

from onegov.activity import AttendeeCollection, Attendee, Booking
from onegov.core.security import Secret
from onegov.feriennet import FeriennetApp, _
from onegov.feriennet.exports.base import FeriennetExport
from onegov.feriennet.forms import PeriodSelectForm
from onegov.form import merge_forms
from onegov.org.forms import ExportForm
from sqlalchemy import and_, not_


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Iterator
    from onegov.activity.models import Period, PeriodMeta
    from sqlalchemy.orm import Query, Session


@FeriennetApp.export(
    id='unlucky',
    form_class=merge_forms(ExportForm, PeriodSelectForm),
    permission=Secret,
    title=_('Attendees without Bookings'),
    explanation=_('Attendees who were not granted any of their wishes'),
)
[docs] class UnluckyExport(FeriennetExport):
[docs] def run( self, form: PeriodSelectForm, # type:ignore[override] session: Session ) -> Iterator[Iterator[tuple[str, Any]]]: assert form.selected_period is not None return self.rows(session, form.selected_period)
[docs] def query( self, session: Session, period: Period | PeriodMeta ) -> Query[Attendee]: with_any_bookings = session.query(Booking.attendee_id).filter( Booking.period_id == period.id).subquery() with_accepted_bookings = session.query(Booking.attendee_id).filter( Booking.state == 'accepted', Booking.period_id == period.id).subquery() return AttendeeCollection(session).query().filter( and_( Attendee.id.in_(with_any_bookings), not_(Attendee.id.in_(with_accepted_bookings)), ))
[docs] def rows( self, session: Session, period: Period | PeriodMeta ) -> Iterator[Iterator[tuple[str, Any]]]: for user in self.query(session, period): yield ((k, v) for k, v in self.fields(user))
[docs] def fields(self, attendee: Attendee) -> Iterator[tuple[str, Any]]: yield from self.attendee_fields(attendee) yield from self.user_fields(attendee.user)