from __future__ import annotations
from onegov.core.collection import GenericCollection
from onegov.core.orm.abstract import MoveDirection
from onegov.people.models import AgencyMembership
from typing import Literal
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from collections.abc import Iterator
from sqlalchemy.orm import Query
from uuid import UUID
[docs]
class AgencyMembershipCollection(GenericCollection[AgencyMembership]):
""" Manages a list of agency memberships.
Use it like this::
from onegov.people import AgencyMembershipCollection
memberships = AgencyMembershipCollection(session)
"""
@property
[docs]
def model_class(self) -> type[AgencyMembership]:
return AgencyMembership
[docs]
def by_id(self, id: UUID) -> AgencyMembership | None: # type:ignore
return super().query().filter(
self.primary_key == id).first()
[docs]
def query(self, order_by: str | None = None) -> Query[AgencyMembership]:
query = super().query()
if not order_by:
return query
assert hasattr(self.model_class, order_by)
return query.order_by(getattr(self.model_class, order_by))
[docs]
def move(
self,
subject: AgencyMembership,
target: AgencyMembership,
direction: MoveDirection,
move_on_col: Literal['order_within_person', 'order_within_agency']
) -> None:
""" Takes the given subject and moves it somewhere in relation to the
target.
:subject:
The item to be moved.
:target:
The item above which or below which the subject is moved.
:direction:
The direction relative to the target. Either
:attr:`MoveDirection.above` if the subject should be moved
above the target, or :attr:`MoveDirection.below` if the subject
should be moved below the target.
:move_on_col:
Designates the column for which the new order should be evaluated.
Possible values are `order_within_agency` and
`order_within_person`.
"""
assert isinstance(target, self.model_class)
assert isinstance(subject, self.model_class)
assert hasattr(subject, move_on_col)
assert hasattr(target, move_on_col)
if move_on_col == 'order_within_person':
siblings = target.siblings_by_person.all()
else:
siblings = target.siblings_by_agency.all()
def new_order() -> Iterator[AgencyMembership]:
for sibling in siblings:
if sibling == subject:
continue
if sibling == target and direction == MoveDirection.above:
yield subject
yield target
continue
if sibling == target and direction == MoveDirection.below:
yield target
yield subject
continue
yield sibling
for order, sibling in enumerate(new_order()):
setattr(sibling, move_on_col, order)