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