Source code for election_day.models.election_compound.election_compound

from __future__ import annotations

from onegov.core.orm import Base, observes
from onegov.core.orm import translation_hybrid
from onegov.core.orm.mixins import ContentMixin
from onegov.core.orm.mixins import dict_property
from onegov.core.orm.mixins import meta_property
from onegov.core.orm.types import HSTORE
from onegov.core.utils import groupbylist
from onegov.election_day.models.election_compound.mixins import (
    DerivedAttributesMixin)
from onegov.election_day.models.mixins import DomainOfInfluenceMixin
from onegov.election_day.models.mixins import ExplanationsPdfMixin
from onegov.election_day.models.mixins import IdFromTitlesMixin
from onegov.election_day.models.mixins import LastModifiedMixin
from onegov.election_day.models.mixins import TitleTranslationsMixin
from onegov.election_day.models.party_result.mixins import (
    HistoricalPartyResultsMixin)
from onegov.election_day.models.party_result.mixins import (
    PartyResultsCheckMixin)
from onegov.election_day.models.party_result.mixins import (
    PartyResultsOptionsMixin)
from onegov.file import NamedFile
from sqlalchemy import Column, Boolean
from sqlalchemy import Date
from sqlalchemy import Text
from sqlalchemy.orm import object_session
from sqlalchemy.orm import relationship


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Collection
    from collections.abc import Mapping
    from onegov.core.types import AppenderQuery
    from onegov.election_day.models import Election
    from onegov.election_day.models import ElectionCompoundRelationship
    from onegov.election_day.models import Notification
    from onegov.election_day.models import PartyPanachageResult
    from onegov.election_day.models import PartyResult
    from onegov.election_day.models import Screen
    from onegov.election_day.types import DomainOfInfluence
    import datetime

[docs] rel = relationship
[docs] class ElectionCompound( Base, ContentMixin, LastModifiedMixin, DomainOfInfluenceMixin, TitleTranslationsMixin, IdFromTitlesMixin, PartyResultsOptionsMixin, PartyResultsCheckMixin, HistoricalPartyResultsMixin, ExplanationsPdfMixin, DerivedAttributesMixin ):
[docs] __tablename__ = 'election_compounds'
@property
[docs] def polymorphic_base(self) -> type[ElectionCompound]: return ElectionCompound
#: Identifies the election compound, may be used in the url
[docs] id: Column[str] = Column(Text, primary_key=True)
#: external identifier
[docs] external_id: Column[str | None] = Column(Text, nullable=True)
#: all translations of the title
[docs] title_translations: Column[Mapping[str, str]] = Column( HSTORE, nullable=False )
#: the translated title (uses the locale of the request, falls back to the #: default locale of the app)
[docs] title = translation_hybrid(title_translations)
#: all translations of the short title
[docs] short_title_translations: Column[Mapping[str, str] | None] = Column( HSTORE, nullable=True )
#: the translated short title (uses the locale of the request, falls back #: to the default locale of the app)
[docs] short_title = translation_hybrid(short_title_translations)
@observes('title_translations', 'short_title_translations')
[docs] def title_observer( self, title_translations: Mapping[str, str], short_title_translations: Mapping[str, str] ) -> None: if not self.id: self.id = self.id_from_title(object_session(self))
#: Shortcode for cantons that use it
[docs] shortcode: Column[str | None] = Column(Text, nullable=True)
#: The date of the elections
[docs] date: Column[datetime.date] = Column(Date, nullable=False)
#: Doppelter Pukelsheim
[docs] pukelsheim: Column[bool] = Column(Boolean, nullable=False, default=False)
#: Allow setting the status of the compound and its elections manually
[docs] completes_manually: Column[bool] = Column( Boolean, nullable=False, default=False )
#: Status of the compound and its elections
[docs] manually_completed: Column[bool] = Column( Boolean, nullable=False, default=False )
#: An election compound may contains n party results
[docs] party_results: relationship[list[PartyResult]] = relationship( 'PartyResult', cascade='all, delete-orphan', back_populates='election_compound' )
#: An election compound may contains n party panachage results
[docs] party_panachage_results: rel[list[PartyPanachageResult]]
party_panachage_results = relationship( 'PartyPanachageResult', cascade='all, delete-orphan', back_populates='election_compound' ) #: An election compound may have related election compounds
[docs] related_compounds: rel[AppenderQuery[ElectionCompoundRelationship]]
related_compounds = relationship( 'ElectionCompoundRelationship', foreign_keys='ElectionCompoundRelationship.source_id', cascade='all, delete-orphan', back_populates='source', lazy='dynamic' ) #: An election compound may be related by other election compounds
[docs] referencing_compounds: rel[AppenderQuery[ElectionCompoundRelationship]]
referencing_compounds = relationship( 'ElectionCompoundRelationship', foreign_keys='ElectionCompoundRelationship.target_id', cascade='all, delete-orphan', back_populates='target', lazy='dynamic' ) #: Defines optional colors for parties
[docs] colors: dict_property[dict[str, str]] = meta_property( 'colors', default=dict )
#: Defines the domain of the elections
[docs] domain_elections: dict_property[DomainOfInfluence] = meta_property( 'domain_elections', default='district' )
#: An election compound may contain n elections
[docs] elections: relationship[list[Election]] = relationship( 'Election', cascade='all', back_populates='election_compound', order_by='Election.shortcode' )
@observes('elections')
[docs] def elections_observer( self, elections: Collection[Election] ) -> None: changes = {c for e in elections if (c := e.last_result_change)} if changes: new = max(changes) old = self.last_result_change if not old or (old and old < new): self.last_result_change = new
@property
[docs] def progress(self) -> tuple[int, int]: """ Returns a tuple with the current progress. If the elections define a `domain_supersegment` (i.e. superregions), this is the number of fully counted supersegments vs. the total number of supersegments. If no `domain_supersegment` is defined, this is the number of counted elections vs. the total number of elections. """ pairs = sorted( (e.domain_supersegment, e.completed) for e in self.elections ) grouped = groupbylist(pairs, lambda x: x[0]) if len(grouped) == 1 and grouped[0][0] == '': result = [completed for _, completed in grouped[0][1]] else: result = [all(c for _, c in segment) for _, segment in grouped] return sum(1 for r in result if r), len(result)
@property
[docs] def has_results(self) -> bool: """ Returns True, if the election compound has any results. """ if self.has_party_results: return True if self.has_party_panachage_results: return True for election in self.elections: if election.has_results: return True return False
@property
[docs] def elected_candidates(self) -> list[tuple[str, str]]: """ Returns the first and last names of the elected candidates. """ result = [] for election in self.elections: result.extend(election.elected_candidates) return result
#: notifcations linked to this election compound
[docs] notifications: relationship[AppenderQuery[Notification]]
notifications = relationship( # type:ignore[misc] 'onegov.election_day.models.notification.Notification', back_populates='election_compound', lazy='dynamic' ) #: screens linked to this election compound
[docs] screens: relationship[AppenderQuery[Screen]] = relationship( 'Screen', back_populates='election_compound', )
#: may be used to store a link related to this election #: additional file in case of Doppelter Pukelsheim
[docs] upper_apportionment_pdf = NamedFile()
#: additional file in case of Doppelter Pukelsheim
[docs] lower_apportionment_pdf = NamedFile()
@property
[docs] def relationships_for_historical_party_results( self ) -> AppenderQuery[ElectionCompoundRelationship]: return self.related_compounds
[docs] def clear_results(self, clear_all: bool = False) -> None: """ Clears all related results. """ self.last_result_change = None self.party_results = [] self.party_panachage_results = [] for election in self.elections: election.clear_results(clear_all)