from __future__ import annotations
from onegov.activity.models.invoice_item import InvoiceItem, SCALE
from onegov.activity.models.period import Period
from onegov.core.orm import Base
from onegov.core.orm.mixins import TimestampMixin
from onegov.core.orm.types import UUID
from onegov.pay import Price
from onegov.user import User
from sqlalchemy import and_
from sqlalchemy import Column
from sqlalchemy import ForeignKey
from sqlalchemy import func
from sqlalchemy import select
from sqlalchemy.ext.hybrid import hybrid_property
from sqlalchemy.orm import object_session, relationship, joinedload
from uuid import uuid4
from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
import uuid
from collections.abc import Iterable
from decimal import Decimal
from onegov.activity.models import InvoiceReference
from sqlalchemy.sql import ColumnElement
[docs]
def sync_invoice_items(
items: Iterable[InvoiceItem],
capture: bool = True
) -> None:
for item in items:
if not item.payments:
continue
if capture:
for payment in item.payments:
# though it should be fairly rare, it's possible for
# charges not to be captured yet
if payment.state == 'open':
# FIXME: This only works for StripePayment, should we
# check that we got a StripePayment?
payment.charge.capture() # type:ignore[attr-defined]
payment.sync()
# the last payment is the relevant one
item.paid = item.payments[-1].state == 'paid'
[docs]
class Invoice(Base, TimestampMixin):
""" A grouping of invoice items. """
[docs]
__tablename__ = 'invoices'
#: the public id of the invoice
[docs]
id: Column[uuid.UUID] = Column(
UUID, # type:ignore[arg-type]
primary_key=True,
default=uuid4
)
#: the period to which this invoice belongs to
[docs]
period_id: Column[uuid.UUID] = Column(
UUID, # type:ignore[arg-type]
ForeignKey('periods.id'),
nullable=False
)
[docs]
period: relationship[Period] = relationship(
Period,
back_populates='invoices'
)
#: the user to which the invoice belongs
[docs]
user_id: Column[uuid.UUID] = Column(
UUID, # type:ignore[arg-type]
ForeignKey('users.id'),
nullable=False
)
# FIXME: Do we need this backref? It's across module boundaries, so
# not the best for proper module isolation
[docs]
user: relationship[User] = relationship(User, backref='invoices')
#: the specific items linked with this invoice
[docs]
items: relationship[list[InvoiceItem]] = relationship(
InvoiceItem,
back_populates='invoice'
)
#: the references pointing to this invoice
[docs]
references: relationship[list[InvoiceReference]] = relationship(
'InvoiceReference',
back_populates='invoice',
cascade='all, delete-orphan'
)
@property
[docs]
def price(self) -> Price:
return Price(self.outstanding_amount, 'CHF')
@property
[docs]
def has_donation(self) -> bool:
for item in self.items:
if item.group == 'donation':
return True
return False
[docs]
def readable_by_bucket(self, bucket: str) -> str | None:
for ref in self.references:
if ref.bucket == bucket:
return ref.readable
return None
[docs]
def sync(self, capture: bool = True) -> None:
items = object_session(self).query(InvoiceItem).filter(and_(
InvoiceItem.source != None,
InvoiceItem.source != 'xml'
)).options(joinedload(InvoiceItem.payments))
sync_invoice_items(items, capture=capture)
[docs]
def add(
self,
group: str,
text: str,
unit: Decimal,
quantity: Decimal,
organizer: str = '',
attendee_id: uuid.UUID | None = None,
flush: bool = True,
**kwargs: Any # FIXME: type safety for optional arguments
) -> InvoiceItem:
item = InvoiceItem(
group=group,
attendee_id=attendee_id,
text=text,
organizer=organizer,
unit=unit,
quantity=quantity,
invoice_id=self.id,
**kwargs
)
self.items.append(item)
if flush:
object_session(self).flush()
return item
@hybrid_property
[docs]
def discourage_changes(self) -> bool:
return self.discourage_changes_for_items(self.items)
@hybrid_property
[docs]
def disable_changes(self) -> bool:
return self.disable_changes_for_items(self.items)
@hybrid_property
[docs]
def has_online_payments(self) -> bool:
return self.has_online_payments_for_items(self.items)
[docs]
def discourage_changes_for_items(
self,
items: Iterable[InvoiceItem]
) -> bool:
for item in items:
if item.source == 'xml':
return True
return False
[docs]
def disable_changes_for_items(
self,
items: Iterable[InvoiceItem]
) -> bool:
for item in items:
if not item.source:
continue
if item.source == 'xml':
continue
states = {p.state for p in item.payments}
if 'open' in states or 'paid' in states:
return True
return False
[docs]
def has_online_payments_for_items(
self,
items: Iterable[InvoiceItem]
) -> bool:
for item in items:
if not item.source or item.source == 'xml':
continue
return True
return False
if TYPE_CHECKING:
total_amount: Column[Decimal]
outstanding_amount: Column[Decimal]
paid_amount: Column[Decimal]
# paid or not
@hybrid_property # type:ignore[no-redef]
def paid(self) -> bool:
return self.outstanding_amount <= 0
# paid + unpaid
@hybrid_property # type:ignore[no-redef]
[docs]
def total_amount(self) -> Decimal:
return self.outstanding_amount + self.paid_amount
@total_amount.expression # type:ignore[no-redef]
def total_amount(cls) -> ColumnElement[Decimal]:
return select([func.sum(InvoiceItem.amount)]).where(
InvoiceItem.invoice_id == cls.id
).label('total_amount')
# paid only
@hybrid_property # type:ignore[no-redef]
[docs]
def outstanding_amount(self):
return round(
sum(item.amount for item in self.items if not item.paid),
SCALE
)
@outstanding_amount.expression # type:ignore[no-redef]
def outstanding_amount(cls):
return select([func.sum(InvoiceItem.amount)]).where(
and_(
InvoiceItem.invoice_id == cls.id,
InvoiceItem.paid == False
)
).label('outstanding_amount')
# unpaid only
@hybrid_property # type:ignore[no-redef]
[docs]
def paid_amount(self) -> Decimal:
return round(
sum(item.amount for item in self.items if item.paid),
SCALE
)
@paid_amount.expression # type:ignore[no-redef]
def paid_amount(cls) -> ColumnElement[Decimal]:
return select([func.sum(InvoiceItem.amount)]).where(
and_(
InvoiceItem.invoice_id == cls.id,
InvoiceItem.paid == True
)
).label('paid_amount')