Source code for swissvotes.fields.metadata
from __future__ import annotations
from decimal import Decimal
from onegov.form.fields import UploadField
from onegov.form.validators import FileSizeLimit
from onegov.form.validators import WhitelistedMimeType
from onegov.swissvotes import _
from onegov.swissvotes.models import ColumnMapperMetadata
from openpyxl import load_workbook
from wtforms.validators import ValidationError
from typing import Any
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from collections.abc import Sequence
from onegov.core.types import FileDict as StrictFileDict
from onegov.form.types import FormT
from onegov.form.types import Filter
from onegov.form.types import PricingRules
from onegov.form.types import Validators
from onegov.form.types import Widget
from typing import Self
from wtforms.form import BaseForm
from wtforms.meta import _SupportsGettextAndNgettext
from wtforms.meta import DefaultMeta
[docs]
class SwissvoteMetadataField(UploadField):
""" An upload field expecting Swissvotes metadata (XLSX). """
if TYPE_CHECKING:
def __init__(
self,
label: str | None = None,
validators: Validators[FormT, Self] | None = None,
filters: Sequence[Filter] = (),
description: str = '',
id: str | None = None,
default: Sequence[StrictFileDict] = (),
widget: Widget[Self] | None = None,
render_kw: dict[str, Any] | None = None,
name: str | None = None,
_form: BaseForm | None = None,
_prefix: str = '',
_translations: _SupportsGettextAndNgettext | None = None,
_meta: DefaultMeta | None = None,
# onegov specific kwargs that get popped off
*,
fieldset: str | None = None,
depends_on: Sequence[Any] | None = None,
pricing: PricingRules | None = None,
) -> None: ...
else:
def __init__(self, *args, **kwargs):
kwargs.setdefault('validators', [])
kwargs['validators'].append(
WhitelistedMimeType({
'application/excel',
'application/octet-stream',
'application/vnd.ms-excel',
'application/vnd.ms-office',
(
'application/vnd.openxmlformats-officedocument'
'.spreadsheetml.sheet'
),
'application/zip'
})
)
kwargs['validators'].append(FileSizeLimit(10 * 1024 * 1024))
kwargs.setdefault('render_kw', {})
kwargs['render_kw']['force_simple'] = True
super().__init__(*args, **kwargs)
[docs]
def post_validate(
self,
form: BaseForm,
validation_stopped: bool
) -> None:
""" Make sure the given XLSX is valid (all expected columns are
present all cells contain reasonable values).
Converts the XLSX to a list of metadata dictionaries objects,
available as ``data``.
"""
super().post_validate(form, validation_stopped)
if validation_stopped:
return
assert self.file is not None
errors = []
data: dict[Decimal, dict[str, Any]] = {}
mapper = ColumnMapperMetadata()
try:
workbook = load_workbook(self.file, data_only=True)
except Exception as exception:
raise ValidationError(_('Not a valid XLSX file.')) from exception
if len(workbook.worksheets) < 1:
raise ValidationError(_('No data.'))
if 'Metadaten zu Scans' not in workbook.sheetnames:
raise ValidationError(_("Sheet 'Metadaten zu Scans' is missing."))
sheet = workbook['Metadaten zu Scans']
if TYPE_CHECKING:
from openpyxl.worksheet.worksheet import Worksheet
assert isinstance(sheet, Worksheet)
if sheet.max_row <= 1:
raise ValidationError(_('No data.'))
headers = [column.value for column in next(sheet.rows)]
missing = set(mapper.columns.values()) - set(headers) # type:ignore
if missing:
raise ValidationError(_(
'Some columns are missing: ${columns}.',
mapping={'columns': ', '.join(missing)}
))
value: Any | None
for index in range(2, sheet.max_row + 1):
metadata: dict[str, Any] = {}
all_columns_empty = True
column_errors = []
for (
attribute, column, type_, nullable, precision, scale
) in mapper.items():
cell = sheet.cell(index, headers.index(column) + 1)
try:
if cell.value is None:
value = None
elif type_ == 'TEXT':
if (
cell.data_type == 'n'
and int(cell.value) == cell.value # type:ignore
):
value = str(int(cell.value)) # type:ignore
else:
value = str(cell.value)
value = '' if value == '.' else value
elif type_ == 'INTEGER':
if cell.data_type == 's':
value = cell.value
value = '' if value == '.' else value
value = int(
value # type:ignore[arg-type]
) if value else None
else:
value = int(cell.value) # type:ignore[arg-type]
elif type_ and type_.startswith('NUMERIC'):
if isinstance(cell.value, str):
value = cell.value
value = '' if value == '.' else value
value = Decimal(str(value)) if value else None
else:
value = Decimal(str(cell.value))
if value is not None:
value = Decimal(
format(value, f'{precision}.{scale}f')
)
all_columns_empty = all_columns_empty and value is None
except Exception:
column_errors.append((
index, column, f"'{value}' ≠ {type_ and type_.lower()}"
))
else:
if not nullable and value is None:
column_errors.append((index, column, '∅'))
mapper.set_value(metadata, attribute, value)
if not all_columns_empty:
errors.extend(column_errors)
if not column_errors:
bfs_number = metadata['bfs_number']
filename = metadata['filename']
data.setdefault(bfs_number, {})[filename] = metadata
if errors:
raise ValidationError(_(
'Some cells contain invalid values: ${errors}.',
mapping={
'errors': '; '.join(
'{}:{} {}'.format(*error) for error in errors
)
}
))
self.data = data