from __future__ import annotations
from morepath.request import Response
from onegov.core.security import Private
from onegov.core.security import Public
from onegov.core.security import Secret
from onegov.form import Form
from onegov.swissvotes import _
from onegov.swissvotes import SwissvotesApp
from onegov.swissvotes.collections import SwissVoteCollection
from onegov.swissvotes.external_resources import MfgPosters
from onegov.swissvotes.external_resources import BsPosters
from onegov.swissvotes.external_resources import SaPosters
from onegov.swissvotes.forms import SearchForm
from onegov.swissvotes.forms import UpdateDatasetForm
from onegov.swissvotes.forms import UpdateExternalResourcesForm
from onegov.swissvotes.forms import UpdateMetadataForm
from onegov.swissvotes.layouts import DeleteVotesLayout
from onegov.swissvotes.layouts import UpdateExternalResourcesLayout
from onegov.swissvotes.layouts import UpdateMetadataLayout
from onegov.swissvotes.layouts import UpdateVotesLayout
from onegov.swissvotes.layouts import VotesLayout
from translationstring import TranslationString
from typing import TYPE_CHECKING
if TYPE_CHECKING:
from onegov.core.types import RenderData
from onegov.swissvotes.request import SwissvotesRequest
from webob import Response as BaseResponse
@SwissvotesApp.form(
model=SwissVoteCollection,
permission=Public,
form=SearchForm,
template='votes.pt'
)
[docs]
def view_votes(
self: SwissVoteCollection,
request: SwissvotesRequest,
form: SearchForm
) -> RenderData:
form.submitted(request)
if not form.errors:
form.apply_model(self)
return {
'layout': VotesLayout(self, request),
'form': form
}
@SwissvotesApp.form(
model=SwissVoteCollection,
permission=Private,
form=UpdateDatasetForm,
template='form.pt',
name='update'
)
[docs]
def update_votes(
self: SwissVoteCollection,
request: SwissvotesRequest,
form: UpdateDatasetForm
) -> RenderData | BaseResponse:
self = self.default()
layout = UpdateVotesLayout(self, request)
if form.submitted(request):
added, updated = self.update(form.dataset.data)
request.message(
_(
'Dataset updated (${added} added, ${updated} updated)',
mapping={'added': added, 'updated': updated}
),
'success'
)
# Warn if descriptor labels are missing
missing = set()
for vote in self.query():
for policy_area in vote.policy_areas:
missing |= {
path for path in policy_area.label_path
if not isinstance(path, TranslationString)
}
if missing:
request.message(
_(
'The dataset contains unknown descriptors: ${items}.',
mapping={'items': ', '.join(sorted(missing))}
),
'warning'
)
return request.redirect(layout.votes_url)
return {
'layout': layout,
'form': form,
'cancel': request.link(self),
'button_text': _('Update'),
}
@SwissvotesApp.form(
model=SwissVoteCollection,
permission=Private,
form=UpdateMetadataForm,
template='form.pt',
name='update-metadata'
)
@SwissvotesApp.form(
model=SwissVoteCollection,
permission=Private,
form=UpdateExternalResourcesForm,
template='form.pt',
name='update-external-resources'
)
[docs]
def update_external_resources(
self: SwissVoteCollection,
request: SwissvotesRequest,
form: UpdateExternalResourcesForm
) -> RenderData | BaseResponse:
self = self.default()
layout = UpdateExternalResourcesLayout(self, request)
if form.submitted(request):
added_total = 0
updated_total = 0
removed_total = 0
failed_total = set()
source: SaPosters | BsPosters | MfgPosters
for resource in form.resources.data or ():
if resource == 'sa':
source = SaPosters()
elif resource == 'bs':
assert request.app.bs_api_token is not None
source = BsPosters(request.app.bs_api_token)
elif resource == 'mfg':
assert request.app.mfg_api_token is not None
source = MfgPosters(request.app.mfg_api_token)
else:
raise AssertionError('unreachable')
added, updated, removed, failed = source.fetch(request.session)
added_total += added
updated_total += updated
removed_total += removed
failed_total |= failed
request.message(
_(
'External resources updated (${added} added, '
'${updated} updated, ${removed} removed)',
mapping={
'added': added_total,
'updated': updated_total,
'removed': removed_total
}
),
'success'
)
if failed_total:
failed_total_str = ', '.join(
layout.format_bfs_number(item[0]) for item in sorted(
failed_total)
)
request.message(
_(
'Some external resources could not be updated: ${failed}',
mapping={'failed': failed_total_str}
),
'warning'
)
details = ', '.join(
f'[{layout.format_bfs_number(item[0])}: obj {item[1]}]'
for item in sorted(failed_total)
)
request.message(
_(
'Details ${details}', mapping={'details': details}
),
'warning'
)
return request.redirect(layout.votes_url)
return {
'layout': layout,
'form': form,
'cancel': request.link(self),
'button_text': _('Update external sources for images'),
}
@SwissvotesApp.view(
model=SwissVoteCollection,
permission=Public,
name='csv'
)
[docs]
def export_votes_csv(
self: SwissVoteCollection,
request: SwissvotesRequest
) -> Response:
return Response(
request.app.get_cached_dataset('csv'),
content_type='text/csv',
content_disposition='inline; filename=dataset.csv'
)
@SwissvotesApp.view(
model=SwissVoteCollection,
permission=Public,
name='xlsx'
)
[docs]
def export_votes_xlsx(
self: SwissVoteCollection,
request: SwissvotesRequest
) -> Response:
return Response(
request.app.get_cached_dataset('xlsx'),
content_type=(
'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'
),
content_disposition='inline; filename=dataset.xlsx'
)
@SwissvotesApp.form(
model=SwissVoteCollection,
permission=Secret,
form=Form,
template='form.pt',
name='delete'
)
[docs]
def delete_votes(
self: SwissVoteCollection,
request: SwissvotesRequest,
form: Form
) -> RenderData | BaseResponse:
self = self.default()
layout = DeleteVotesLayout(self, request)
if form.submitted(request):
for vote in self.query():
request.session.delete(vote)
request.message(_('All votes deleted'), 'success')
return request.redirect(layout.votes_url)
return {
'layout': layout,
'form': form,
'message': _('Do you really want to delete all votes?!'),
'button_text': _('Delete'),
'button_class': 'alert',
'cancel': request.link(self)
}