Source code for org.forms.event

from __future__ import annotations

import json
import transaction

from functools import cached_property
from datetime import date, datetime, timedelta
from dateutil import rrule
from dateutil.parser import parse
from dateutil.rrule import rrulestr
from itertools import chain

from onegov.core.csv import convert_excel_to_csv
from onegov.core.csv import CSVFile
from onegov.event.collections import EventCollection, OccurrenceCollection
from onegov.event.models import EventFile
from onegov.form import Form
from onegov.form.fields import MultiCheckboxField
from onegov.form.fields import TimeField
from onegov.form.fields import UploadField
from onegov.form.fields import UploadFileWithORMSupport
from onegov.form.utils import get_fields_from_class
from onegov.form.validators import (
    FileSizeLimit, ValidPhoneNumber, ValidFilterFormDefinition)
from onegov.form.validators import WhitelistedMimeType
from onegov.gis import CoordinatesField
from onegov.org import _
from onegov.ticket import TicketCollection
from sedate import replace_timezone, to_timezone
from wtforms.fields import BooleanField
from wtforms.fields import DateField
from wtforms.fields import EmailField
from wtforms.fields import RadioField
from wtforms.fields import StringField
from wtforms.fields import TextAreaField
from wtforms.validators import DataRequired
from wtforms.validators import Email
from wtforms.validators import InputRequired
from wtforms.validators import Optional


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Sequence
    from onegov.core.csv import DefaultRow
    from onegov.event.models import Event, Occurrence
    from onegov.org.request import OrgRequest


[docs] TAGS = [ _('Art'), _('Cinema'), _('Concert'), _('Congress'), _('Culture'), _('Dancing'), _('Education'), _('Exhibition'), _('Gastronomy'), _('Health'), _('Library'), _('Literature'), _('Market'), _('Meetup'), _('Misc'), _('Music School'), _('Nature'), _('Music'), _('Party'), _('Politics'), _('Reading'), _('Religion'), _('Sports'), _('Talk'), _('Theater'), _('Tourism'), _('Toy Library'), _('Tradition'), _('Youth'), _('Elderly'), ]
[docs] WEEKDAYS = ( ('MO', _('Mo')), ('TU', _('Tu')), ('WE', _('We')), ('TH', _('Th')), ('FR', _('Fr')), ('SA', _('Sa')), ('SU', _('Su')), )
[docs] class EventForm(Form): """ Defines the form for all events. """ if TYPE_CHECKING:
[docs] request: OrgRequest
[docs] timezone = 'Europe/Zurich'
[docs] email = EmailField( label=_('Submitter'), description='max.muster@example.org', validators=[InputRequired(), Email()] )
[docs] title = StringField( label=_('Title'), description=_('Concerto in the castle garden'), validators=[InputRequired()] )
[docs] description = TextAreaField( label=_('Description'), description=_('Enjoy a concerto in the castle garden.'), render_kw={'rows': 12} )
[docs] image = UploadFileWithORMSupport( label=_('Image'), file_class=EventFile, validators=[ Optional(), WhitelistedMimeType({ 'image/gif', 'image/jpeg', 'image/png' }), FileSizeLimit(5 * 1024 * 1024) ] )
[docs] pdf = UploadFileWithORMSupport( label=_('Additional Information (PDF)'), file_class=EventFile, validators=[ Optional(), WhitelistedMimeType({ 'application/pdf', }), FileSizeLimit(5 * 1024 * 1024) ] )
[docs] location = StringField( label=_('Venue'), description='Pilatusstrasse 3, 6000 Luzern', validators=[InputRequired()] )
[docs] price = TextAreaField( label=_('Price'), description=_('10 CHF for adults'), render_kw={'rows': 2} )
[docs] organizer = StringField( label=_('Organizer'), description=_('Music society'), validators=[InputRequired()] )
[docs] organizer_email = EmailField( label=_('Organizer E-Mail address'), description=_('Shown as contact E-Mail address'), validators=[Optional(), Email()] )
[docs] organizer_phone = StringField( label=_('Organizer phone number'), description=_('Shown as contact phone number'), validators=[Optional(), ValidPhoneNumber()] )
[docs] external_event_url = StringField( label=_('External event URL'), description='https://www.example.ch', )
[docs] event_registration_url = StringField( label=_('Event Registration URL'), description='/form/event-registration, https://www.example.ch', )
[docs] coordinates = CoordinatesField( label=_('Coordinates'), description=_('The marker can be moved by dragging it with the mouse'), render_kw={'data-map-type': 'marker'} )
[docs] tags = MultiCheckboxField( label=_('Tags'), choices=[(tag, tag) for tag in TAGS], )
[docs] start_date = DateField( label=_('Date'), validators=[InputRequired()], default=date.today )
[docs] start_time = TimeField( label=_('From'), description='18:00', validators=[InputRequired()] )
[docs] end_time = TimeField( label=_('To'), description='19:15', validators=[InputRequired()] )
[docs] repeat = RadioField( label=_('Repeat'), default='without', choices=( ('without', _('Without')), ('weekly', _('Weekly')), ('dates', _('On additional dates')) ) )
[docs] weekly = MultiCheckboxField( label=_('Repeats itself every'), choices=WEEKDAYS, render_kw={'prefix_label': False, 'class_': 'oneline-checkboxes'}, depends_on=('repeat', 'weekly') )
[docs] end_date = DateField( label=_('Until date'), validators=[Optional()], depends_on=('repeat', 'weekly') )
[docs] dates = TextAreaField( label=_('Dates'), depends_on=('repeat', 'dates'), render_kw={'class_': 'many many-dates'}, )
@property
[docs] def start(self) -> datetime: assert self.start_date.data is not None assert self.start_time.data is not None return replace_timezone( datetime( self.start_date.data.year, self.start_date.data.month, self.start_date.data.day, self.start_time.data.hour, self.start_time.data.minute ), self.timezone )
@property
[docs] def end(self) -> datetime: end_date = self.start_date.data assert end_date is not None assert self.start_time.data is not None assert self.end_time.data is not None if self.end_time.data <= self.start_time.data: end_date += timedelta(days=1) return replace_timezone( datetime( end_date.year, end_date.month, end_date.day, self.end_time.data.hour, self.end_time.data.minute ), self.timezone )
def __init__(self, *args: Any, **kwargs: Any) -> None: super().__init__(*args, **kwargs)
[docs] self.date_errors: dict[int, str] = {}
[docs] def populate_submitter(self) -> None: if self.request.is_logged_in: self.email.data = self.request.current_username
[docs] def on_request(self) -> None: if self.tags: if self.custom_tags(): self.tags.choices = [(tag, tag) for tag in self.custom_tags() or ()] self.sort_tags() if not self.dates.data: self.dates.data = self.dates_to_json(None) if not self.email.data: self.populate_submitter()
[docs] def custom_tags(self) -> list[str] | None: return self.request.app.custom_event_tags
[docs] def sort_tags(self) -> None: assert isinstance(self.tags.choices, list) self.tags.choices.sort(key=lambda c: self.request.translate(c[1]))
[docs] def ensure_start_before_end(self) -> bool | None: if self.start_date.data and self.end_date.data: if self.start_date.data > self.end_date.data: assert isinstance(self.end_date.errors, list) self.end_date.errors.append( _('The end date must be later than the start date.') ) return False return None
[docs] def ensure_valid_rrule(self) -> bool | None: """ Make sure a valid RRULE can be generated with the given fields. Might be better to group weekly and end_date in an enclosure, see See http://wtforms.readthedocs.org/en/latest/fields.html #field-enclosures. """ if self.repeat.data == 'weekly': # FIXME: the logic is a bit wacky here, partially because we # allow omitting both weekly and end in which case we # leave the model's recurrence unchanged, this doesn't # seem sound to me, why don't we just make both weekly # and end_date InputRequired? Then a lot of this logic # goes away too and we just have to check whether the # weekday matches the start date... result = None if self.weekly.data and self.start_date.data: weekday = WEEKDAYS[self.start_date.data.weekday()][0] if weekday not in self.weekly.data: assert isinstance(self.weekly.errors, list) self.weekly.errors.append( _('The weekday of the start date must be selected.') ) result = False if self.weekly.data and not self.end_date.data: assert isinstance(self.end_date.errors, list) self.end_date.errors.append( _('Please set and end date if the event is recurring.') ) result = False elif self.end_date.data and not self.weekly.data: assert isinstance(self.weekly.errors, list) self.weekly.errors.append( _('Please select a weekday if the event is recurring.') ) result = False return result if self.repeat.data == 'dates': try: assert self.json_to_dates(self.dates.data) except (AssertionError, ValueError): assert isinstance(self.repeat.errors, list) self.repeat.errors.append(_('Invalid dates.')) return False return None
[docs] def populate_obj(self, model: Event) -> None: # type:ignore[override] """ Stores the form values on the model. """ super().populate_obj(model, exclude={ 'start_date', 'start_time', 'end_date', 'end_time', 'weekly', 'email', }) # clear the recurrence to avoid updating all occurrences too much model.recurrence = '' model.meta = model.meta or {} model.meta['submitter_email'] = self.email.data model.timezone = self.timezone model.start = self.start model.end = self.end if self.repeat.data == 'without': self.recurrence = None elif self.repeat.data == 'weekly': if self.weekly.data and self.end_date.data: assert self.end_time.data is not None until_date = to_timezone( replace_timezone( datetime( self.end_date.data.year, self.end_date.data.month, self.end_date.data.day, self.end_time.data.hour, self.end_time.data.minute ), model.timezone ), 'UTC' ) model.recurrence = ( 'RRULE:FREQ=WEEKLY;WKST=MO;BYDAY={};UNTIL={}'.format( ','.join(self.weekly.data), until_date.strftime('%Y%m%dT%H%M%SZ') ) ) elif self.repeat.data == 'dates': model.recurrence = '\n'.join( f"RDATE:{d.strftime('%Y%m%dT000000Z')}" for d in self.parsed_dates ) else: raise NotImplementedError if self.request.app.org.event_filter_type in ('filters', 'tags_and_filters'): filter_keywords = {} for field in self.request.app.org.event_filter_fields: form_field = getattr(self, field.id) filter_keywords[field.id] = form_field.data if filter_keywords: model.filter_keywords = filter_keywords for occ in model.occurrences: occ.filter_keywords = filter_keywords
[docs] def process_obj(self, model: Event) -> None: # type:ignore[override] """ Stores the page values on the form. """ super().process_obj(model) self.start_time.data = model.localized_start.time() self.end_time.data = model.localized_end.time() self.start_date.data = model.localized_start.date() if model.recurrence: last_occurrence = model.occurrence_dates(localize=True)[-1] self.end_date.data = last_occurrence.date() rule = rrulestr(model.recurrence) if getattr(rule, '_freq', None) == rrule.WEEKLY: self.repeat.data = 'weekly' self.dates.data = self.dates_to_json(None) self.weekly.data = [ WEEKDAYS[day][0] # FIXME: can we access this via public API? for day in rule._byweekday # type:ignore[union-attr] ] else: self.repeat.data = 'dates' self.dates.data = self.dates_to_json(tuple(rule)) elif self.errors.get('weekly'): self.dates.data = self.dates_to_json(None) self.repeat.data = 'weekly' else: self.dates.data = self.dates_to_json(None) self.repeat.data = 'without' if model.meta: self.email.data = model.meta.get('submitter_email') if model.filter_keywords: keywords = model.filter_keywords for field in self.request.app.org.event_filter_fields: form_field = getattr(self, field.id, None) if form_field is None: continue form_field.data = keywords.get(field.id, None)
@cached_property
[docs] def parsed_dates(self) -> list[date]: return self.json_to_dates(self.dates.data)
[docs] def json_to_dates(self, text: str | None = None) -> list[date]: if not text: return [] return [ date.fromisoformat(value['date']) for value in json.loads(text).get('values', []) ]
[docs] def dates_to_json(self, dates: Sequence[date] | None = None) -> str: dates = dates or [] return json.dumps({ 'labels': { 'date': self.request.translate(_('Date')), 'add': self.request.translate(_('Add')), 'remove': self.request.translate(_('Remove')), }, 'values': [ { 'date': d.isoformat(), 'error': self.date_errors.get(ix, '') } for ix, d in enumerate(dates) ] })
[docs] class EventImportForm(Form): if TYPE_CHECKING:
[docs] request: OrgRequest
[docs] clear = BooleanField( label=_('Clear'), description=_( 'Delete imported events before importing. This does not delete ' 'otherwise imported events and submitted events.' ), default=False )
[docs] dry_run = BooleanField( label=_('Dry Run'), description=_('Do not actually import the events.'), default=False )
[docs] file = UploadField( label=_('Import'), validators=[ DataRequired(), WhitelistedMimeType({ 'application/excel', 'application/vnd.ms-excel', ( 'application/' 'vnd.openxmlformats-officedocument.spreadsheetml.sheet' ), 'application/vnd.ms-office', 'application/octet-stream', 'application/zip', 'text/csv', 'text/plain', }), FileSizeLimit(10 * 1024 * 1024) ], render_kw={'force_simple': True} )
@property
[docs] def headers(self) -> dict[str, str]: return { 'title': self.request.translate(_('Title')), 'description': self.request.translate(_('Description')), 'location': self.request.translate(_('Venue')), 'price': self.request.translate(_('Price')), 'organizer': self.request.translate(_('Organizer')), 'organizer_email': self.request.translate(_('Organizer E-Mail ' 'address')), 'organizer_phone': self.request.translate(_('Organizer phone ' 'number')), 'external_event_url': self.request.translate( _('External event URL')), 'event_registration_url': self.request.translate( _('Event Registration URL')), 'tags': self.request.translate(_('Tags')), 'start': self.request.translate(_('From')), 'end': self.request.translate(_('To')), }
[docs] def custom_tags(self) -> list[str] | None: return self.request.app.custom_event_tags
[docs] def run_export(self) -> list[dict[str, str]]: occurrences = OccurrenceCollection(self.request.session) headers = self.headers def get(occurrence: Occurrence, attribute: str) -> str: if attribute in ('start', 'end'): attribute = f'localized_{attribute}' result = ( getattr(occurrence, attribute, None) or getattr(occurrence.event, attribute, None) ) if isinstance(result, datetime): result = result.strftime('%d.%m.%Y %H:%M') if attribute == 'tags': result = ', '.join( self.request.translate(_(tag)) for tag in result or () ) result = result or '' result = result.strip() return result return [ { title: get(occurrence, attribute) for attribute, title in headers.items() } for occurrence in occurrences.query() ]
[docs] def run_import(self) -> tuple[int, list[str]]: headers = self.headers session = self.request.session events = EventCollection(session) all_tags = chain( TAGS, self.custom_tags() or [] ) tags = { self.request.translate(tag): tag for tag in all_tags } tickets = TicketCollection(session) if self.clear.data: for event in events.query(): if event.source: continue if tickets.by_handler_id(event.id.hex): continue session.delete(event) assert self.file.file is not None csvfile = convert_excel_to_csv(self.file.file) try: csv = CSVFile(csvfile, expected_headers=headers.values()) except Exception: error_string = self.request.translate( _('Expected header line with the following columns:') ) return 0, [f'0 - {error_string} {list(headers.values())}'] lines = list(csv.lines) columns = { key: csv.as_valid_identifier(value) for key, value in headers.items() } def get(line: DefaultRow, column: str, attribute: str) -> Any: result = getattr(line, column) if attribute in ('start', 'end'): result = parse(result, dayfirst=True) if attribute == 'tags': result = result.split(', ') result = [tags.get(tag, None) for tag in result] result = [tag for tag in result if tag] return result count = 0 errors = [] for number, line in enumerate(lines, start=1): try: kwargs = { attribute: get(line, column, attribute) for attribute, column in columns.items() } kwargs['timezone'] = 'Europe/Zurich' event = events.add(**kwargs) event.meta['submitter_email'] = self.request.current_username event.submit() event.publish() count += 1 except Exception: errors.append(str(number)) if self.dry_run.data or errors: transaction.abort() return count, errors
[docs] class EventConfigurationForm(Form): """ Form to configure filters for events view. """
[docs] definition = TextAreaField( label=_('Definition'), fieldset=_('General'), validators=[ InputRequired(), ValidFilterFormDefinition( require_email_field=False, require_title_fields=False, reserved_fields={name for name, _ in get_fields_from_class(EventForm)} ) ], render_kw={'rows': 32, 'data-editor': 'form'})
[docs] keyword_fields = TextAreaField( label=_('Filters'), fieldset=_('Display'), render_kw={ 'class_': 'formcode-select', 'data-fields-include': 'radio,checkbox' })