Source code for reservation.collection

from __future__ import annotations

import enum
from onegov.core.utils import normalize_for_url
from onegov.reservation.models import Resource
from uuid import uuid4, UUID


from typing import overload, Any, Literal, TypeVar, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Callable
    from libres.context.core import Context
    from libres.db.models import Allocation, Reservation
    from sqlalchemy.orm import Query
    from typing import TypeAlias


[docs] _R = TypeVar('_R', bound=Resource)
[docs] class _Marker(enum.Enum):
[docs] any_type = enum.auto()
[docs] any_type_t: TypeAlias = Literal[_Marker.any_type] # noqa: PYI042
[docs] any_type: any_type_t = _Marker.any_type
[docs] class ResourceCollection: """ Manages a list of resources. """ def __init__(self, libres_context: Context): assert hasattr(libres_context, 'get_service'), """ The ResourceCollection expected the libres_contex, not the session. """
[docs] self.libres_context = libres_context
[docs] self.session = libres_context.get_service('session_provider').session()
[docs] def query(self) -> Query[Resource]: return self.session.query(Resource)
[docs] def add( self, title: str, timezone: str, type: str | None = None, name: str | None = None, meta: dict[str, Any] | None = None, content: dict[str, Any] | None = None, definition: str | None = None, group: str | None = None ) -> Resource: if type is None: resource = Resource() else: # look up the right class depending on the type (we need to do # this a bit akwardly here, because Resource does not use the # ModelBase as declarative base) resource = Resource.get_polymorphic_class(type, Resource)() resource.id = uuid4() resource.name = name or normalize_for_url(title) resource.title = title resource.timezone = timezone resource.meta = meta or {} resource.content = content or {} resource.definition = definition resource.group = group resource.renew_access_token() self.session.add(resource) self.session.flush() return self.bind(resource)
@overload
[docs] def bind(self, resource: _R) -> _R: ...
@overload def bind(self, resource: None) -> None: ... def bind(self, resource: _R | None) -> _R | None: if resource: resource.bind_to_libres_context(self.libres_context) return resource
[docs] def by_id( self, id: UUID, ensure_type: str | any_type_t = any_type ) -> Resource | None: query = self.query().filter(Resource.id == id) if ensure_type is not any_type: query = query.filter(Resource.type == ensure_type) return self.bind(query.first())
[docs] def by_name( self, name: str, ensure_type: str | any_type_t = any_type ) -> Resource | None: query = self.query().filter(Resource.name == name) if ensure_type is not any_type: query = query.filter(Resource.type == ensure_type) return self.bind(query.first())
[docs] def by_allocation(self, allocation: Allocation) -> Resource | None: return self.by_id(allocation.resource)
[docs] def by_reservation(self, reservation: Reservation) -> Resource | None: return self.by_id(reservation.resource)
[docs] def delete( self, resource: Resource, including_reservations: bool = False, handle_reservation: Callable[[Reservation], Any] | None = None ) -> None: scheduler = resource.get_scheduler(self.libres_context) if not including_reservations: assert not scheduler.managed_reserved_slots().first() assert not scheduler.managed_reservations().first() scheduler.managed_allocations().delete('fetch') else: if callable(handle_reservation): for res in scheduler.managed_reservations(): # e.g. create a ticket snapshot handle_reservation(res) scheduler.extinguish_managed_records() if resource.files: # unlink any linked files resource.files = [] self.session.flush() self.session.delete(resource) self.session.flush()