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)