Source code for election_day.models.election.list

from __future__ import annotations

from onegov.core.orm import Base
from onegov.core.orm.mixins import TimestampMixin
from onegov.core.orm.types import UUID
from onegov.election_day.models.election.election_result import ElectionResult
from onegov.election_day.models.election.list_result import ListResult
from onegov.election_day.models.mixins import summarized_property
from sqlalchemy import Column
from sqlalchemy import ForeignKey
from sqlalchemy import func
from sqlalchemy import Integer
from sqlalchemy import select
from sqlalchemy import Text
from sqlalchemy.orm import object_session
from sqlalchemy.orm import relationship
from uuid import uuid4


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    import uuid
    from onegov.election_day.models import Candidate
    from onegov.election_day.models import CandidatePanachageResult
    from onegov.election_day.models import ListConnection
    from onegov.election_day.models import ListPanachageResult
    from onegov.election_day.models import ProporzElection
    from onegov.election_day.types import DistrictPercentage
    from onegov.election_day.types import EntityPercentage
    from sqlalchemy.sql import ColumnElement


[docs] class List(Base, TimestampMixin): """ A list. """
[docs] __tablename__ = 'lists'
#: internal id of the list
[docs] id: Column[uuid.UUID] = Column( UUID, # type:ignore[arg-type] primary_key=True, default=uuid4 )
#: external id of the list
[docs] list_id: Column[str] = Column(Text, nullable=False)
# number of mandates
[docs] number_of_mandates: Column[int] = Column( Integer, nullable=False, default=lambda: 0 )
#: name of the list
[docs] name: Column[str] = Column(Text, nullable=False)
#: the election id this list belongs to
[docs] election_id: Column[str] = Column( Text, ForeignKey('elections.id', onupdate='CASCADE', ondelete='CASCADE'), nullable=False )
#: the election this list belongs to
[docs] election: relationship[ProporzElection] = relationship( 'ProporzElection', back_populates='lists' )
#: the list connection id this list belongs to
[docs] connection_id: Column[uuid.UUID | None] = Column( UUID, # type:ignore[arg-type] ForeignKey('list_connections.id', ondelete='CASCADE'), nullable=True )
#: the list connection this list belongs to
[docs] connection: relationship[ListConnection] = relationship( 'ListConnection', back_populates='lists' )
#: a list contains n candidates
[docs] candidates: relationship[list[Candidate]] = relationship( 'Candidate', cascade='all, delete-orphan', back_populates='list', )
#: a list contains n results
[docs] results: relationship[list[ListResult]] = relationship( 'ListResult', cascade='all, delete-orphan', back_populates='list', )
#: a list contains additional votes from other lists
[docs] panachage_results: relationship[list[ListPanachageResult]]
panachage_results = relationship( 'ListPanachageResult', foreign_keys='ListPanachageResult.target_id', cascade='all, delete-orphan', back_populates='target' ) #: a list contains to other lists lost votes
[docs] panachage_results_lost: relationship[list[ListPanachageResult]]
panachage_results_lost = relationship( 'ListPanachageResult', foreign_keys='ListPanachageResult.source_id', cascade='all, delete-orphan', back_populates='source' ) #: an list contains n (outgoing) candidate panachage results
[docs] candidate_panachage_results: relationship[list[CandidatePanachageResult]]
candidate_panachage_results = relationship( 'CandidatePanachageResult', cascade='all, delete-orphan', back_populates='list' ) #: the total votes
[docs] votes = summarized_property('votes')
[docs] def aggregate_results(self, attribute: str) -> int: """ Gets the sum of the given attribute from the results. """ return sum(getattr(result, attribute) for result in self.results)
@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(ListResult, attribute)), 0 ) ]) expr = expr.where(ListResult.list_id == cls.id) return expr.label(attribute)
@property
[docs] def percentage_by_entity(self) -> dict[int, EntityPercentage]: """ Returns the percentage of votes by the entity. Includes uncounted entities and entities with no results available. """ query = self.election.results_query query = query.join(ElectionResult.list_results) query = query.filter(ListResult.list_id == self.id) totals_by_entity = self.election.votes_by_entity.subquery() results_sub = query.with_entities( ElectionResult.entity_id.label('id'), ElectionResult.counted.label('counted'), ListResult.votes.label('votes') ).subquery() session = object_session(self) results = session.query( results_sub.c.id, results_sub.c.counted, totals_by_entity.c.votes.label('total'), results_sub.c.votes ) results = results.join( totals_by_entity, totals_by_entity.c.entity_id == results_sub.c.id ) percentage: dict[int, EntityPercentage] = { r.id: { 'counted': r.counted, 'votes': r.votes, 'percentage': round( 100 * (r.votes / r.total), 2) if r.total else 0.0 } for r in results } empty = self.election.results_query.with_entities( ElectionResult.entity_id.label('id'), ElectionResult.counted.label('counted') ) empty = empty.filter( ElectionResult.entity_id.notin_([r.id for r in results]) ) percentage.update({ r.id: { 'counted': r.counted, 'percentage': 0.0, 'votes': 0 } for r in empty} ) return percentage
@property
[docs] def percentage_by_district(self) -> dict[str, DistrictPercentage]: """ Returns the percentage of votes aggregated by the distict. Includes uncounted districts and districts with no results available. """ query = self.election.results_query.order_by(None) query = query.join(ElectionResult.list_results) query = query.filter(ListResult.list_id == self.id) totals_by_district = self.election.votes_by_district.subquery() query = query.with_entities( ElectionResult.district.label('name'), func.sum(ListResult.votes).label('votes'), ) query = query.group_by(ElectionResult.district) results_sub = query.subquery() session = object_session(self) results = session.query( results_sub.c.name, totals_by_district.c.entities, totals_by_district.c.counted, totals_by_district.c.votes.label('total'), results_sub.c.votes ) results = results.join( totals_by_district, totals_by_district.c.district == results_sub.c.name ) percentage: dict[str, DistrictPercentage] = { r.name: { 'counted': r.counted, 'entities': r.entities, 'votes': r.votes, 'percentage': round( 100 * (r.votes / r.total), 2) if r.total else 0.0 } for r in results } empty = self.election.results_query.with_entities( ElectionResult.district.label('name'), func.array_agg(ElectionResult.entity_id).label('entities'), func.coalesce( func.bool_and(ElectionResult.counted), False ).label('counted') ) empty = empty.group_by(ElectionResult.district) empty = empty.order_by(None) for result in empty: update = ( result.name not in percentage or ( set(percentage[result.name]['entities']) != set(result.entities) ) ) if update: percentage[result.name] = { 'counted': result.counted, 'entities': result.entities, 'percentage': 0.0, 'votes': 0 } return percentage