Source code for election_day.formats.imports.election.internal_majorz

from __future__ import annotations

from onegov.election_day import _
from onegov.election_day.formats.imports.common import EXPATS
from onegov.election_day.formats.imports.common import FileImportError
from onegov.election_day.formats.imports.common import get_entity_and_district
from onegov.election_day.formats.imports.common import load_csv
from onegov.election_day.formats.imports.common import STATI
from onegov.election_day.formats.imports.common import validate_color
from onegov.election_day.formats.imports.common import validate_gender
from onegov.election_day.formats.imports.common import validate_integer
from onegov.election_day.models import Candidate
from onegov.election_day.models import CandidateResult
from onegov.election_day.models import ElectionResult
from sqlalchemy.orm import object_session
from uuid import uuid4


from typing import Any
from typing import IO
from typing import TYPE_CHECKING
if TYPE_CHECKING:
    from onegov.core.csv import DefaultRow
    from onegov.election_day.models import Canton
    from onegov.election_day.models import Election
    from onegov.election_day.models import Municipality
    from onegov.election_day.types import Status

    # TODO: Define TypedDict for the parsed results, so we can verify
    #       our parser ensures correct types


[docs] INTERNAL_MAJORZ_HEADERS = ( 'election_status', 'entity_id', 'entity_counted', 'entity_eligible_voters', 'entity_received_ballots', 'entity_blank_ballots', 'entity_invalid_ballots', 'entity_blank_votes', 'entity_invalid_votes', 'candidate_family_name', 'candidate_first_name', 'candidate_id', 'candidate_elected', 'candidate_votes', 'candidate_party', 'election_absolute_majority', )
[docs] def parse_election( line: DefaultRow, errors: list[str] ) -> tuple[int | None, Status | None]: majority = None status = None try: if line.election_absolute_majority: majority = validate_integer(line, 'election_absolute_majority') majority = majority if majority else None status = line.election_status or 'unknown' except ValueError as e: errors.append(e.args[0]) except Exception: errors.append(_('Invalid election values')) if status not in STATI: errors.append(_('Invalid status')) return majority, status # type:ignore[return-value]
[docs] def parse_election_result( line: DefaultRow, errors: list[str], entities: dict[int, dict[str, str]], election: Election, principal: Canton | Municipality ) -> dict[str, Any] | None: try: entity_id = validate_integer(line, 'entity_id') counted = line.entity_counted.strip().lower() == 'true' eligible_voters = validate_integer(line, 'entity_eligible_voters') expats = validate_integer( line, 'entity_expats', optional=True, default=None ) received_ballots = validate_integer(line, 'entity_received_ballots') blank_ballots = validate_integer(line, 'entity_blank_ballots') invalid_ballots = validate_integer(line, 'entity_invalid_ballots') blank_votes = validate_integer(line, 'entity_blank_votes') invalid_votes = validate_integer(line, 'entity_invalid_votes') except ValueError as e: errors.append(e.args[0]) except Exception: errors.append(_('Invalid entity values')) else: if entity_id not in entities and entity_id in EXPATS: entity_id = 0 if entity_id and entity_id not in entities: errors.append(_( '${name} is unknown', mapping={'name': entity_id} )) else: name, district, superregion = get_entity_and_district( entity_id, entities, election, principal, errors ) if not errors: return { 'id': uuid4(), 'election_id': election.id, 'name': name, 'district': district, 'superregion': superregion, 'entity_id': entity_id, 'counted': counted, 'eligible_voters': eligible_voters if counted else 0, 'expats': expats if counted else 0, 'received_ballots': received_ballots if counted else 0, 'blank_ballots': blank_ballots if counted else 0, 'invalid_ballots': invalid_ballots if counted else 0, 'blank_votes': blank_votes if counted else 0, 'invalid_votes': invalid_votes if counted else 0, } return None
[docs] def parse_candidate( line: DefaultRow, errors: list[str], election_id: str, colors: dict[str, str] ) -> dict[str, Any] | None: try: id = validate_integer(line, 'candidate_id') family_name = line.candidate_family_name first_name = line.candidate_first_name elected = str(line.candidate_elected or '').lower() == 'true' party = line.candidate_party color = validate_color(line, 'candidate_party_color') gender = validate_gender(line) year_of_birth = validate_integer( line, 'candidate_year_of_birth', optional=True, default=None ) except ValueError as e: errors.append(e.args[0]) except Exception: errors.append(_('Invalid candidate values')) else: if party and color: colors[party] = color return { 'id': uuid4(), 'candidate_id': id, 'election_id': election_id, 'family_name': family_name, 'first_name': first_name, 'elected': elected, 'party': party, 'gender': gender, 'year_of_birth': year_of_birth } return None
[docs] def parse_candidate_result( line: DefaultRow, errors: list[str], counted: bool ) -> dict[str, Any] | None: try: votes = validate_integer(line, 'candidate_votes') except ValueError as e: errors.append(e.args[0]) else: return { 'id': uuid4(), 'votes': votes if counted else 0, } return None
[docs] def import_election_internal_majorz( election: Election, principal: Canton | Municipality, file: IO[bytes], mimetype: str ) -> list[FileImportError]: """ Tries to import the given file (internal format). This function is typically called automatically every few minutes during an election day - we use bulk inserts to speed up the import. :return: A list containing errors. """ filename = _('Results') csv, error = load_csv( file, mimetype, expected_headers=INTERNAL_MAJORZ_HEADERS, filename=filename, dialect='excel' ) if error is not None: return [error] assert csv is not None errors: list[FileImportError] = [] candidates: dict[str, dict[str, Any]] = {} candidate_results = [] results: dict[int, dict[str, Any]] = {} entities = principal.entities[election.date.year] election_id = election.id colors = election.colors.copy() # This format has one candiate per entity per line absolute_majority = None status = None for line in csv.lines: line_errors: list[str] = [] # Parse the line absolute_majority, status = parse_election(line, line_errors) result = parse_election_result( line, line_errors, entities, election, principal ) counted = (result or {}).get('counted', False) candidate = parse_candidate(line, line_errors, election_id, colors) candidate_result = parse_candidate_result(line, line_errors, counted) # Skip expats if not enabled if result and result['entity_id'] == 0 and not election.has_expats: continue # Pass the errors and continue to the next line if line_errors: errors.extend( FileImportError( error=err, line=line.rownumber, filename=filename ) for err in line_errors ) continue # Add the data assert result is not None result = results.setdefault(result['entity_id'], result) assert candidate is not None assert candidate_result is not None candidate = candidates.setdefault(candidate['candidate_id'], candidate) candidate_result['candidate_id'] = candidate['id'] candidate_result['election_result_id'] = result['id'] candidate_results.append(candidate_result) if not errors and not results: errors.append(FileImportError(_('No data found'))) if errors: return errors # Add the missing entities remaining = set(entities.keys()) if election.has_expats: remaining.add(0) remaining -= set(results.keys()) for entity_id in remaining: name, district, superregion = get_entity_and_district( entity_id, entities, election, principal ) if election.domain == 'none': continue if election.domain == 'municipality': if principal.domain != 'municipality': if name != election.domain_segment: continue if election.domain in ('region', 'district'): if district != election.domain_segment: continue results[entity_id] = { 'id': uuid4(), 'election_id': election_id, 'name': name, 'district': district, 'superregion': superregion, 'entity_id': entity_id, 'counted': False } # Add the results to the DB election.clear_results(True) election.last_result_change = election.timestamp() election.absolute_majority = absolute_majority election.status = status election.colors = colors session = object_session(election) session.bulk_insert_mappings(Candidate, candidates.values()) session.bulk_insert_mappings(ElectionResult, results.values()) session.bulk_insert_mappings(CandidateResult, candidate_results) session.flush() session.expire_all() return []