Source code for activity.collections.occasion

from __future__ import annotations

from decimal import Decimal
from onegov.activity.models import Occasion, OccasionDate
from onegov.activity.types import BoundedIntegerRange
from onegov.core.collection import GenericCollection
from sedate import standardize_date


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Sequence
    from datetime import datetime
    from onegov.activity.models import Activity, Period


[docs] class OccasionCollection(GenericCollection[Occasion]): @property
[docs] def model_class(self) -> type[Occasion]: return Occasion
@staticmethod
[docs] def to_half_open_interval(lower: int, upper: int) -> BoundedIntegerRange: """ Postgres coerces ranges internally to be half-open in an effort to canonize these ranges. This function does the same by taking a closed interval and turning it into a half-open interval of the NumericRange type. """ return BoundedIntegerRange(lower, upper + 1, bounds='[)')
[docs] def add( # type:ignore[override] self, activity: Activity, period: Period, start: datetime, end: datetime, timezone: str, meeting_point: str | None = None, age: Sequence[int] | None = None, spots: Sequence[int] | None = None, note: str | None = None, cost: Decimal = Decimal(0), exclude_from_overlap_check: bool = False ) -> Occasion: occasion = super().add( meeting_point=meeting_point, age=age and self.to_half_open_interval(*age), spots=spots and self.to_half_open_interval(*spots), note=note, activity_id=activity.id, period_id=period.id, cost=cost, exclude_from_overlap_check=exclude_from_overlap_check ) self.add_date(occasion, start, end, timezone) return occasion
[docs] def add_date( self, occasion: Occasion, start: datetime, end: datetime, timezone: str ) -> OccasionDate: start = standardize_date(start, timezone) end = standardize_date(end, timezone) date = OccasionDate( start=start, end=end, timezone=timezone ) occasion.dates.append(date) self.session.flush() return date
[docs] def find_date( self, occasion: Occasion, start: datetime, end: datetime, timezone: str ) -> OccasionDate | None: start = standardize_date(start, timezone) end = standardize_date(end, timezone) q = self.session.query(OccasionDate) q = q.filter(OccasionDate.start == start) q = q.filter(OccasionDate.end == end) q = q.filter(OccasionDate.timezone == timezone) return q.first()
[docs] def remove_date(self, date: OccasionDate) -> None: q = self.session.query(OccasionDate) q = q.filter(OccasionDate.id == date.id) occasion_date = q.first() if occasion_date: self.session.delete(occasion_date)
[docs] def clear_dates(self, occasion: Occasion) -> None: q = self.session.query(OccasionDate) q = q.filter(OccasionDate.occasion_id == occasion.id) for date in q: self.session.delete(date) occasion.dates = [] self.session.flush()