Source code for election_day.models.vote.vote

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.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 StatusMixin
from onegov.election_day.models.mixins import summarized_property
from onegov.election_day.models.mixins import TitleTranslationsMixin
from onegov.election_day.models.vote.ballot import Ballot
from onegov.election_day.models.vote.mixins import DerivedBallotsCountMixin
from sqlalchemy import Column
from sqlalchemy import Date
from sqlalchemy import func
from sqlalchemy import select
from sqlalchemy import Text
from sqlalchemy.ext.hybrid import hybrid_property
from sqlalchemy.orm import object_session
from sqlalchemy.orm import relationship
from uuid import uuid4


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    import datetime
    from collections.abc import Mapping
    from onegov.core.types import AppenderQuery
    from onegov.election_day.models import DataSourceItem
    from onegov.election_day.models import Notification
    from onegov.election_day.models import Screen
    from onegov.election_day.types import BallotType
    from sqlalchemy.sql import ColumnElement


[docs] class Vote( Base, ContentMixin, LastModifiedMixin, DomainOfInfluenceMixin, StatusMixin, TitleTranslationsMixin, IdFromTitlesMixin, DerivedBallotsCountMixin, ExplanationsPdfMixin ): """ A vote describes the issue being voted on. For example, "Vote for Net Neutrality" or "Vote for Basic Income". """
[docs] __tablename__ = 'votes'
@property
[docs] def polymorphic_base(self) -> type[Vote]: return Vote
#: the type of the item, this can be used to create custom polymorphic #: subclasses of this class. See #: `<https://docs.sqlalchemy.org/en/improve_toc/\ #: orm/extensions/declarative/inheritance.html>`_.
[docs] type: Column[str] = Column(Text, nullable=False)
[docs] __mapper_args__ = { 'polymorphic_on': type, 'polymorphic_identity': 'simple' }
#: identifies the vote, may be used in the url, generated from the title
[docs] id: Column[str] = Column(Text, primary_key=True)
#: external identifier
[docs] external_id: Column[str | None] = Column(Text, nullable=True)
#: shortcode for cantons that use it
[docs] shortcode: 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))
#: identifies the date of the vote
[docs] date: Column[datetime.date] = Column(Date, nullable=False)
#: a vote contains n ballots
[docs] ballots: relationship[list[Ballot]] = relationship( 'Ballot', cascade='all, delete-orphan', order_by='Ballot.type', lazy='joined', back_populates='vote' )
[docs] def ballot( self, ballot_type: BallotType ) -> Ballot: """ Returns the given ballot if it exists, creates it if not. """ result = None for ballot in self.ballots: if ballot.type == ballot_type: result = ballot break if not result: result = Ballot(id=uuid4(), type=ballot_type) self.ballots.append(result) return result
@property
[docs] def proposal(self) -> Ballot: return self.ballot('proposal')
@property
[docs] def counted(self) -> bool: # type:ignore[override] """ Checks if there are results for all entities. """ if not self.ballots: return False for ballot in self.ballots: if not ballot.counted: return False return True
@property
[docs] def has_results(self) -> bool: """ Returns True, if there are any results. """ for ballot in self.ballots: for result in ballot.results: if result.counted: return True return False
@property
[docs] def answer(self) -> str | None: if not self.counted or not self.proposal: return None if self.tie_breaker_vocabulary: return 'proposal' if self.proposal.accepted else 'counter-proposal' return 'accepted' if self.proposal.accepted else 'rejected'
@property
[docs] def yeas_percentage(self) -> float: """ The percentage of yeas (discounts empty/invalid ballots). """ return self.yeas / ((self.yeas + self.nays) or 1) * 100
@property
[docs] def nays_percentage(self) -> float: """ The percentage of nays (discounts empty/invalid ballots). """ return 100 - self.yeas_percentage
@property
[docs] def progress(self) -> tuple[int, int]: """ Returns a tuple with the first value being the number of counted entities and the second value being the number of total entities. For complex votes, it is assumed that every ballot has the same progress. """ if not self.proposal or not self.proposal.results: return 0, 0 return ( sum(1 for result in self.proposal.results if result.counted), len(self.proposal.results) )
@property
[docs] def counted_entities(self) -> list[str]: """ Returns the names of the already counted entities. Might contain an empty string in case of expats. For complex votes, it is assumed that every ballot has the same progress. """ if not self.proposal or not self.proposal.results: return [] return sorted( result.name for result in self.proposal.results if result.counted )
#: the total yeas
[docs] yeas = summarized_property('yeas')
#: the total nays
[docs] nays = summarized_property('nays')
#: the total empty votes
[docs] empty = summarized_property('empty')
#: the total invalid votes
[docs] invalid = summarized_property('invalid')
#: the total eligible voters
[docs] eligible_voters = summarized_property('eligible_voters')
#: the total expats
[docs] expats = summarized_property('expats')
[docs] def aggregate_results(self, attribute: str) -> int: """ Gets the sum of the given attribute from the results. """ return sum(getattr(ballot, attribute) for ballot in self.ballots)
@classmethod
[docs] def aggregate_results_expression( cls, attribute: str ) -> ColumnElement[int]: """ Gets the sum of the given attribute from the results, as SQL expression. """ expr = select([ func.coalesce( func.sum(getattr(Ballot, attribute)), 0 ) ]) expr = expr.where(Ballot.vote_id == cls.id) return expr.label(attribute)
if TYPE_CHECKING:
[docs] last_ballot_change: Column[datetime.datetime | None]
last_modified: Column[datetime.datetime | None] @hybrid_property # type:ignore[no-redef] def last_ballot_change(self) -> datetime.datetime | None: """ Returns last change of the vote, its ballots and any of its results. """ changes = [ change for ballot in self.ballots if (change := ballot.last_change) ] return max(changes) if changes else None @last_ballot_change.expression # type:ignore[no-redef] def last_ballot_change(cls) -> ColumnElement[datetime.datetime | None]: expr = select([func.max(Ballot.last_change)]) expr = expr.where(Ballot.vote_id == cls.id) expr = expr.label('last_ballot_change') return expr @hybrid_property # type:ignore[no-redef]
[docs] def last_modified(self) -> datetime.datetime | None: """ Returns last change of the vote, its ballots and any of its results. """ changes = [ change for ballot in self.ballots if (change := ballot.last_change) ] last_change = self.last_change if last_change is not None: changes.append(last_change) last_result_change = self.last_result_change if last_result_change is not None: changes.append(last_result_change) return max(changes) if changes else None
@last_modified.expression # type:ignore[no-redef] def last_modified(cls) -> ColumnElement[datetime.datetime | None]: return func.greatest( cls.last_change, cls.last_result_change, cls.last_ballot_change ) #: data source items linked to this vote
[docs] data_sources: relationship[list[DataSourceItem]] = relationship( 'DataSourceItem', back_populates='vote' )
#: notifcations linked to this vote
[docs] notifications: relationship[AppenderQuery[Notification]]
notifications = relationship( # type:ignore[misc] 'onegov.election_day.models.notification.Notification', back_populates='vote', lazy='dynamic' ) #: screens linked to this vote
[docs] screens: relationship[AppenderQuery[Screen]] = relationship( 'Screen', back_populates='vote', )
#: may be used to store a link related to this vote #: Additional, translatable label for the link #: may be used to indicate that the vote contains expats as seperate #: resultas (typically with entity_id = 0)
[docs] has_expats: dict_property[bool] = meta_property('expats', default=False)
#: The segment of the domain. This might be the municipality, if this is a #: communal vote.
[docs] domain_segment: dict_property[str] = meta_property( 'domain_segment', default='' )
#: Use the vocabulary of a tie breaker. This is a silly trick introduced #: 2024 by ZG in the course of the transparency initiative and only used #: there - all other principals use a proper complex vote.
[docs] tie_breaker_vocabulary: dict_property[bool] = meta_property( 'tie_breaker_vocabulary', default=False )
#: direct or indirect counter/proposal
[docs] direct: dict_property[bool] = meta_property('direct', default=True)
[docs] def clear_results(self, clear_all: bool = False) -> None: """ Clear all the results. """ self.status = None self.last_result_change = None if clear_all: self.ballots = [] else: for ballot in self.ballots: ballot.clear_results()