Source code for election_day.models.vote.mixins

from __future__ import annotations

from sqlalchemy import case
from sqlalchemy import cast
from sqlalchemy import Float
from sqlalchemy import func
from sqlalchemy.ext.hybrid import hybrid_property


from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from sqlalchemy import Column
    from sqlalchemy.sql import ColumnElement


[docs] class DerivedAttributesMixin: """ A simple mixin to add commonly used functions to ballots and their results. """ if TYPE_CHECKING:
[docs] yeas_percentage: Column[float]
nays_percentage: Column[float] accepted: Column[bool] @hybrid_property # type: ignore[no-redef] def yeas_percentage(self) -> float: """ The percentage of yeas (discounts empty/invalid ballots). """ return self.yeas / ((self.yeas + self.nays) or 1) * 100 @yeas_percentage.expression # type:ignore[no-redef] def yeas_percentage(cls) -> ColumnElement[float]: # coalesce will pick the first non-null result # nullif will return null if division by zero # => when all yeas and nays are zero the yeas percentage is 0% return 100 * ( cast(cls.yeas, Float) / cast( func.coalesce( func.nullif(cls.yeas + cls.nays, 0), 1 ), Float ) ) @hybrid_property # type:ignore[no-redef]
[docs] def nays_percentage(self) -> float: """ The percentage of nays (discounts empty/invalid ballots). """ return 100 - self.yeas_percentage
@hybrid_property # type:ignore[no-redef]
[docs] def accepted(self) -> bool: return self.yeas > self.nays if self.counted else None
@accepted.expression # type:ignore[no-redef] def accepted(cls) -> ColumnElement[bool]: return case({True: cls.yeas > cls.nays}, cls.counted)
[docs] class DerivedBallotsCountMixin: """ A simple mixin to add commonly used functions to votes, ballots and their results. """ if TYPE_CHECKING:
[docs] cast_ballots: Column[int]
turnout: Column[float] # forward declare required columns yeas: Column[int] nays: Column[int] empty: Column[int] invalid: Column[int] @hybrid_property # type:ignore[no-redef] def cast_ballots(self) -> int: return ( (self.yeas or 0) + (self.nays or 0) + (self.empty or 0) + (self.invalid or 0) ) @hybrid_property # type:ignore[no-redef]
[docs] def turnout(self) -> float: return ( self.cast_ballots / self.eligible_voters * 100 if self.eligible_voters else 0 )
@turnout.expression # type:ignore[no-redef] def turnout(cls) -> ColumnElement[float]: return case( [( cls.eligible_voters > 0, cast(cls.cast_ballots, Float) / cast(cls.eligible_voters, Float) * 100 )], else_=0 )