Source code for pay.collections.payment

from __future__ import annotations

from collections import defaultdict
from onegov.core.collection import GenericCollection, Pagination
from onegov.pay.models import Payment
from sqlalchemy import desc
from sqlalchemy.orm import joinedload
from sqlalchemy.orm import undefer


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Collection, Iterable
    from datetime import datetime
    from decimal import Decimal
    from onegov.pay.types import AnyPayableBase, PaymentState
    from sqlalchemy.orm import Query, Session
    from typing import Self
    from uuid import UUID


[docs] class PaymentCollection(GenericCollection[Payment], Pagination[Payment]): """ Manages the payment records. To render a list of payments you might want to also consider the :class:`onegov.pay.collection.payable.Paybale` collection, which renders payments by loading the linked records first. """
[docs] page: int
def __init__( self, session: Session, source: str = '*', page: int = 0, start: datetime | None = None, end: datetime | None = None ): GenericCollection.__init__(self, session) Pagination.__init__(self, page)
[docs] self.source = source
[docs] self.start = start
[docs] self.end = end
@property
[docs] def model_class(self) -> type[Payment]: return Payment.get_polymorphic_class(self.source, Payment)
[docs] def add( self, *, source: str | None = None, amount: Decimal | None = None, currency: str = 'CHF', remote_id: str | None = None, state: PaymentState = 'open', # FIXME: We probably don't want to allow arbitrary kwargs # but we need to make sure, we don't use any other # ones somewhere first **kwargs: Any ) -> Payment: if source is None: if self.source == '*': source = 'generic' else: source = self.source return super().add( source=source, amount=amount, currency=currency, remote_id=remote_id, state=state, **kwargs )
[docs] def __eq__(self, other: object) -> bool: if not isinstance(other, PaymentCollection): return False return ( self.source == other.source and self.page == other.page and self.start == other.start and self.end == other.end )
[docs] def subset(self) -> Query[Payment]: q = self.query().order_by(desc(Payment.created)) if self.start: q = q.filter(self.start <= Payment.created) if self.end: q = q.filter(Payment.created <= self.end) q = q.options(joinedload(Payment.provider)) # type:ignore[misc] q = q.options(undefer(Payment.created)) return q
@property
[docs] def page_index(self) -> int: return self.page
[docs] def page_by_index(self, index: int) -> Self: return self.__class__(self.session, self.source, index)