from __future__ import annotations
from decimal import Decimal
from functools import cached_property
from onegov.activity.models import Invoice, InvoiceItem
from onegov.activity.models.invoice import sync_invoice_items
from onegov.activity.models.invoice_reference import KNOWN_SCHEMAS, Schema
from onegov.core.collection import GenericCollection
from sqlalchemy import func, and_, not_
from sqlalchemy.orm import joinedload
from onegov.user import User
from uuid import uuid4, UUID
from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
from collections.abc import Collection
from sqlalchemy.orm import Query, Session
from sqlalchemy.sql import ColumnElement
from typing import Self
[docs]
class InvoiceCollection(GenericCollection[Invoice]):
def __init__(
self,
session: Session,
period_id: UUID | None = None,
user_id: UUID | None = None,
schema: str = 'feriennet-v1',
schema_config: dict[str, Any] | None = None
):
super().__init__(session)
[docs]
self.period_id = period_id
if schema not in KNOWN_SCHEMAS:
raise RuntimeError(f'Unknown schema: {schema}')
[docs]
self.schema_name = schema
[docs]
self.schema_config = (schema_config or {})
@cached_property
[docs]
def schema(self) -> Schema:
return KNOWN_SCHEMAS[self.schema_name](**self.schema_config)
[docs]
def query(self, ignore_period_id: bool = False) -> Query[Invoice]:
q = super().query()
if self.user_id:
q = q.filter_by(user_id=self.user_id)
if self.period_id is not None and not ignore_period_id:
q = q.filter_by(period_id=self.period_id)
return q
[docs]
def query_items(self) -> Query[InvoiceItem]:
return self.session.query(InvoiceItem).filter(
InvoiceItem.invoice_id.in_(
self.query().with_entities(Invoice.id).subquery()
)
)
[docs]
def for_user_id(self, user_id: UUID | None) -> Self:
return self.__class__(self.session, self.period_id, user_id,
self.schema_name, self.schema_config)
[docs]
def for_period_id(self, period_id: UUID | None) -> Self:
return self.__class__(self.session, period_id, self.user_id,
self.schema_name, self.schema_config)
[docs]
def for_schema(
self,
schema: str,
schema_config: dict[str, Any] | None = None
) -> Self:
return self.__class__(self.session, self.period_id, self.user_id,
schema, schema_config)
[docs]
def update_attendee_name(
self,
attendee_id: UUID,
attendee_name: str
) -> None:
invoice_items = self.query_items()
for item in invoice_items:
if item.attendee_id == attendee_id:
item.group = attendee_name
@cached_property
[docs]
def invoice(self) -> str | None:
# XXX used for compatibility with legacy implementation in Feriennet
return self.period_id and self.period_id.hex or None
@cached_property
[docs]
def username(self) -> str | None:
# XXX used for compatibility with legacy implementation in Feriennet
if self.user_id:
user = self.session.query(User).with_entities(
User.username
).filter_by(id=self.user_id).first()
return user and user.username or None
return None
@property
[docs]
def model_class(self) -> type[Invoice]:
return Invoice
[docs]
def _invoice_ids(self) -> Query[tuple[UUID]]:
return self.query().with_entities(Invoice.id).subquery()
[docs]
def _sum(self, condition: ColumnElement[bool]) -> Decimal:
q = self.session.query(func.sum(InvoiceItem.amount).label('amount'))
q = q.filter(condition)
return Decimal(q.scalar() or 0.0)
@property
[docs]
def total_amount(self) -> Decimal:
return self._sum(InvoiceItem.invoice_id.in_(self._invoice_ids()))
@property
[docs]
def outstanding_amount(self) -> Decimal:
return self._sum(and_(
InvoiceItem.invoice_id.in_(self._invoice_ids()),
InvoiceItem.paid == False
))
@property
[docs]
def paid_amount(self) -> Decimal:
return self._sum(and_(
InvoiceItem.invoice_id.in_(self._invoice_ids()),
InvoiceItem.paid == True
))
[docs]
def unpaid_count(
self,
excluded_period_ids: Collection[UUID] | None = None
) -> int:
q = self.query().with_entities(func.count(Invoice.id))
if excluded_period_ids:
q = q.filter(not_(Invoice.period_id.in_(excluded_period_ids)))
q = q.filter(Invoice.paid == False)
return q.scalar() or 0
[docs]
def sync(self) -> None:
items = self.session.query(InvoiceItem).filter(and_(
InvoiceItem.source != None,
InvoiceItem.source != 'xml',
InvoiceItem.invoice_id.in_(
self.query().with_entities(Invoice.id).subquery()
)
)).options(joinedload(InvoiceItem.payments))
sync_invoice_items(items, capture=False)
[docs]
def add( # type:ignore[override]
self,
period_id: UUID | None = None,
user_id: UUID | None = None,
flush: bool = True,
optimistic: bool = False
) -> Invoice:
invoice = Invoice( # type:ignore[misc]
id=uuid4(),
period_id=period_id or self.period_id,
user_id=user_id or self.user_id)
self.session.add(invoice)
self.schema.link(
self.session, invoice, flush=flush, optimistic=optimistic)
if flush:
self.session.flush()
return invoice