Source code for election_day.utils.notification

from __future__ import annotations

from itertools import chain
from onegov.election_day.models.subscriber import Subscriber
from sqlalchemy import and_
from typing import NamedTuple

from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Iterator
    from collections.abc import Sequence
    from onegov.election_day.models import Election
    from onegov.election_day.models import ElectionCompound
    from onegov.election_day.models import Vote
    from sqlalchemy.sql import ColumnElement
    from typing import Literal
    from typing import TypeAlias

[docs] DomainSubset: TypeAlias = Literal['canton', 'municipality'] | None
[docs] class ModelGroup(NamedTuple):
[docs] domain: DomainSubset
[docs] domain_segment: str | None
[docs] elections: Sequence[Election]
[docs] election_compounds: Sequence[ElectionCompound]
[docs] votes: Sequence[Vote]
[docs] filter: ColumnElement[bool]
[docs] def segment_models( elections: Sequence[Election], election_compounds: Sequence[ElectionCompound], votes: Sequence[Vote] ) -> list[ModelGroup]: """ Group elections, compounds and votes by subscribable notification segmenation. """ model_chain: Iterator[Election | ElectionCompound | Vote] model_chain = chain(elections, election_compounds, votes) models = tuple(model_chain) if not models: return [] domains_and_segments: set[tuple[DomainSubset, str | None]] domains_and_segments = { ( 'municipality' if model.domain == 'municipality' else 'canton', getattr(model, 'domain_segment', None) or None ) for model in models } def match_( model: Election | ElectionCompound | Vote, domain: DomainSubset, domain_segment: str | None ) -> bool: if domain != 'municipality': return model.domain != 'municipality' return ( model.domain == 'municipality' and getattr(model, 'domain_segment', None) == domain_segment ) result = [ ModelGroup( domain=None, domain_segment=None, elections=elections, election_compounds=election_compounds, votes=votes, filter=Subscriber.domain.is_(None) ) ] for domain, domain_segment in domains_and_segments: result.append( ModelGroup( domain=domain, domain_segment=domain_segment, elections=[ m for m in elections if match_(m, domain, domain_segment) ], election_compounds=[ m for m in election_compounds if match_(m, domain, domain_segment) ], votes=[m for m in votes if match_(m, domain, domain_segment)], filter=and_( Subscriber.domain == 'municipality', Subscriber.domain_segment == domain_segment ) if domain == 'municipality' else Subscriber.domain != 'municipality', ) ) return result