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()