Source code for feriennet.exports.invoiceitem

from __future__ import annotations

from onegov.activity import (
    Invoice, InvoiceItem, Activity, Occasion, Attendee)
from onegov.core.security import Secret
from onegov.feriennet import FeriennetApp, _
from onegov.feriennet.exports.base import FeriennetExport
from onegov.feriennet.forms import PeriodExportForm
from onegov.user import User
from sqlalchemy.orm import contains_eager
from sqlalchemy import distinct
from sqlalchemy import or_


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


@FeriennetApp.export(
    id='rechnungspositionen',
    form_class=PeriodExportForm,
    permission=Secret,
    title=_('Invoice Items'),
    explanation=_('Exports invoice items in the given period.'),
)
[docs] class InvoiceItemExport(FeriennetExport):
[docs] def run( self, form: PeriodExportForm, # 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 rows( self, session: Session, period: Period ) -> Iterator[Iterator[tuple[str, Any]]]: for item, tags, attendee in self.query(session, period): yield ((k, v) for k, v in self.fields(item, tags, attendee))
[docs] def query( self, session: Session, period: Period ) -> Query[tuple[InvoiceItem, list[str] | None, Attendee]]: # There might be activities with same title from other periods # resulting in double entries of invoice items # We filter by the period in order to get the correct tags activities = session.query( distinct(Activity.title).label('title'), Activity._tags.label('tags') ).join(Occasion).filter( Occasion.period_id == period.id ) activities = activities.subquery() q = session.query(InvoiceItem, activities.c.tags, Attendee) q = q.join(Invoice).join(User) q = q.join( activities, InvoiceItem.text == activities.c.title, isouter=True ).join(Attendee, InvoiceItem.attendee_id == Attendee.id, isouter=True ) q = q.options( contains_eager(InvoiceItem.invoice) .contains_eager(Invoice.user) .undefer(User.data)) q = q.filter(Invoice.period_id == period.id) q = q.filter(or_(User.username == Attendee.username, Attendee.username.is_(None))) q = q.filter(User.id == Invoice.user_id) q = q.order_by( User.username, InvoiceItem.group, InvoiceItem.text ) return q
[docs] def fields( self, item: InvoiceItem, tags: list[str] | None, attendee: Attendee ) -> Iterator[tuple[str, Any]]: yield from self.invoice_item_fields(item) yield from self.user_fields(item.invoice.user) yield from self.activity_tags(tags) yield from self.invoice_attendee_fields(attendee)