Source code for form.collection

from __future__ import annotations

import warnings

from datetime import datetime, timedelta
from onegov.core.crypto import random_token
from onegov.core.utils import normalize_for_url
from onegov.core.collection import GenericCollection
from onegov.file.utils import as_fileintent
from onegov.form.errors import UnableToComplete
from onegov.form.fields import UploadField, UploadMultipleField
from onegov.form.models import (
    FormDefinition,
    FormSubmission,
    FormRegistrationWindow,
    FormFile
)
from sedate import replace_timezone, utcnow
from sqlalchemy import func, exc, inspect
from uuid import uuid4, UUID


from typing import overload, Any, Literal, TYPE_CHECKING

from onegov.form.models.definition import SurveyDefinition
from onegov.form.models.submission import SurveySubmission
from onegov.form.models.survey_window import SurveySubmissionWindow

if TYPE_CHECKING:
    from collections.abc import Callable, Collection, Iterator
    from onegov.form import Form
    from onegov.form.types import SubmissionState
    from onegov.pay.types import PaymentMethod
    from sqlalchemy.orm import Query, Session
    from typing import TypeAlias

[docs] SubmissionHandler: TypeAlias = Callable[[Query[FormSubmission]], Any]
SurveySubmissionHandler: TypeAlias = Callable[[Query[SurveySubmission]], Any] RegistrationWindowHandler: TypeAlias = Callable[ [Query[FormRegistrationWindow]], Any ] SubmissionWindowHandler: TypeAlias = Callable[ [Query[SurveySubmissionWindow]], Any ]
[docs] class FormCollection: """ Manages a collection of forms and form-submissions. """ def __init__(self, session: Session):
[docs] self.session = session
@property
[docs] def definitions(self) -> FormDefinitionCollection: return FormDefinitionCollection(self.session)
@property
[docs] def submissions(self) -> FormSubmissionCollection: return FormSubmissionCollection(self.session)
@property
[docs] def registration_windows(self) -> FormRegistrationWindowCollection: return FormRegistrationWindowCollection(self.session)
@overload
[docs] def scoped_submissions( self, name: str, ensure_existance: Literal[False] ) -> FormSubmissionCollection: ...
@overload def scoped_submissions( self, name: str, ensure_existance: bool = True ) -> FormSubmissionCollection | None: ... def scoped_submissions( self, name: str, ensure_existance: bool = True ) -> FormSubmissionCollection | None: if not ensure_existance or self.definitions.by_name(name): return FormSubmissionCollection(self.session, name) return None # FIXME: This should use Intersection[HasSubmissionsCount] since we # add a temporary attribute to the returned form definitions. # But we have to wait until this feature is available
[docs] def get_definitions_with_submission_count( self ) -> Iterator[FormDefinition]: """ Returns all form definitions and the number of submissions belonging to those definitions, in a single query. The number of submissions is stored on the form definition under the ``submissions_count`` attribute. Only submissions which are 'complete' are considered. """ submissions_ = self.session.query( FormSubmission.name, func.count(FormSubmission.id).label('count') ) submissions_ = submissions_.filter(FormSubmission.state == 'complete') submissions = submissions_.group_by(FormSubmission.name).subquery() definitions = self.session.query(FormDefinition, submissions.c.count) definitions = definitions.outerjoin( submissions, submissions.c.name == FormDefinition.name ) definitions = definitions.order_by(FormDefinition.name) for form, submissions_count in definitions.all(): form.submissions_count = submissions_count or 0 yield form
[docs] class FormDefinitionCollection: """ Manages a collection of forms. """ def __init__(self, session: Session):
[docs] self.session = session
[docs] def query(self) -> Query[FormDefinition]: return self.session.query(FormDefinition)
[docs] def add( self, title: str, definition: str, type: str = 'generic', meta: dict[str, Any] | None = None, content: dict[str, Any] | None = None, name: str | None = None, payment_method: PaymentMethod = 'manual', pick_up: str | None = None ) -> FormDefinition: """ Add the given form to the database. """ # look up the right class depending on the type form = FormDefinition.get_polymorphic_class(type, FormDefinition)() form.name = name or normalize_for_url(title) form.title = title form.definition = definition form.type = type form.meta = meta or {} form.content = content or {} form.payment_method = payment_method form.pick_up = pick_up # try to parse the form (which will throw errors if there are problems) assert form.form_class self.session.add(form) self.session.flush() return form
[docs] def delete( self, name: str, with_submissions: bool = False, with_registration_windows: bool = False, handle_submissions: SubmissionHandler | None = None, handle_registration_windows: RegistrationWindowHandler | None = None, ) -> None: """ Delete the given form. Only possible if there are no submissions associated with it, or if ``with_submissions`` is True. Note that pending submissions are removed automatically, only complete submissions have a bearing on ``with_submissions``. Pass two callbacks to handle additional logic before deleting the objects. """ submissions = self.session.query(FormSubmission) submissions = submissions.filter(FormSubmission.name == name) if not with_submissions: submissions = submissions.filter(FormSubmission.state == 'pending') if handle_submissions: handle_submissions(submissions) # fails if there are linked files in files_for_submissions_files for submission in submissions: for file in submission.files: self.session.delete(file) submissions.delete() if with_registration_windows: registration_windows = self.session.query(FormRegistrationWindow) registration_windows = registration_windows.filter_by(name=name) if handle_registration_windows: handle_registration_windows(registration_windows) registration_windows.delete() self.session.flush() definition = self.by_name(name) if definition: if definition.files: # unlink any linked files before deleting definition.files = [] self.session.flush() # this will fail if there are any submissions left self.query().filter(FormDefinition.name == name).delete('fetch') self.session.flush()
[docs] def by_name(self, name: str) -> FormDefinition | None: """ Returns the given form by name or None. """ return self.query().filter(FormDefinition.name == name).first()
[docs] class FormSubmissionCollection: """ Manages a collection of submissions. """ def __init__(self, session: Session, name: str | None = None):
[docs] self.session = session
[docs] self.name = name
[docs] def query(self) -> Query[FormSubmission]: query = self.session.query(FormSubmission) if self.name is not None: query = query.filter(FormSubmission.name == self.name) return query
[docs] def add( self, name: str | None, form: Form, state: SubmissionState, id: UUID | None = None, payment_method: PaymentMethod | None = None, minimum_price_total: float | None = None, meta: dict[str, Any] | None = None, email: str | None = None, spots: int | None = None ) -> FormSubmission: """ Takes a filled-out form instance and stores the submission in the database. The form instance is expected to have a ``_source`` parameter, which contains the source used to build the form (as only forms with this source may be stored). This method expects the name of the form definition stored in the database. Use :meth:`add_external` to add a submissions whose definition is not stored in the form_definitions table. """ assert hasattr(form, '_source') # this should happen way earlier, we just double check here if state == 'complete': assert form.validate(), "the given form doesn't validate" else: form.validate() if name is None: definition = None else: definition = ( self.session.query(FormDefinition) .filter_by(name=name).one()) if definition is None: registration_window = None else: registration_window = definition.current_registration_window if registration_window: assert spots is not None assert registration_window.accepts_submissions(spots) else: spots = spots or 0 # look up the right class depending on the type submission_class = FormSubmission.get_polymorphic_class( state, FormSubmission ) submission = submission_class() submission.id = id or uuid4() submission.name = name submission.state = state submission.meta = meta or {} submission.email = email submission.registration_window = registration_window submission.spots = spots submission.payment_method = ( payment_method or definition and definition.payment_method or 'manual' ) submission.minimum_price_total = ( minimum_price_total or definition and definition.minimum_price_total or 0.0 ) # extensions are inherited from definitions if definition: assert not submission.extensions, """ For submissions based on definitions, the extensions need to be defined on the definition! """ submission.extensions = definition.extensions self.update(submission, form) self.session.add(submission) self.session.flush() # whenever we add a form submission, we remove all the old ones # which were never completed (this is way easier than having to use # some kind of cronjob ;) self.remove_old_pending_submissions( older_than=utcnow() - timedelta(days=1) ) return submission
[docs] def add_external( self, form: Form, state: SubmissionState, id: UUID | None = None, payment_method: PaymentMethod | None = None, minimum_price_total: float | None = None, meta: dict[str, Any] | None = None, email: str | None = None ) -> FormSubmission: """ Takes a filled-out form instance and stores the submission in the database. The form instance is expected to have a ``_source`` parameter, which contains the source used to build the form (as only forms with this source may be stored). In contrast to :meth:`add`, this method is meant for submissions whose definition is not stored in the form_definitions table. """ return self.add( name=None, form=form, state=state, id=id, payment_method=payment_method, minimum_price_total=minimum_price_total, meta=meta, email=email, )
[docs] def complete_submission(self, submission: FormSubmission) -> None: """ Changes the state to 'complete', if the data is valid. """ assert submission.state == 'pending' if not submission.form_obj.validate(): raise UnableToComplete() submission.state = 'complete' # by completing a submission we are changing it's polymorphic identity, # which is something SQLAlchemy rightly warns us about. Since we know # about it however (and deal with it using self.session.expunge below), # we should ignore this (and only this) warning. with warnings.catch_warnings(): warnings.filterwarnings( action='ignore', message=r'Flushing object', category=exc.SAWarning ) self.session.flush() self.session.expunge(submission)
[docs] def update( self, submission: FormSubmission, form: Form, exclude: Collection[str] | None = None ) -> None: """ Takes a submission and a form and updates the submission data as well as the files stored in a separate table. """ assert submission.id and submission.state # ignore certain fields exclude = set(exclude) if exclude else set() exclude.add(form.meta.csrf_field_name) # never include the csrf token assert hasattr(form, '_source') submission.definition = form._source submission.data = { k: v for k, v in form.data.items() if k not in exclude } submission.update_title(form) # move uploaded files to a separate table files = { field_id for field_id, field in form._fields.items() if isinstance(field, UploadField) and field_id not in exclude # we exclude files that should be removed and submission.data.get(field_id) != {} } files_to_add = { id for id in files if (file_meta := submission.data.get(id)) and not file_meta['data'].startswith('@') } files_to_keep = files - files_to_add multi_files = { field_id: [ index for index, data in enumerate(submission.data.get(field_id, [])) # we exclude files that should be removed but we also if data != {} ] for field_id, field in form._fields.items() if isinstance(field, UploadMultipleField) and field_id not in exclude } multi_files_to_keep = { f'{id}:{idx}' for id, indeces in multi_files.items() if (file_metas := submission.data.get(id)) for idx in indeces # if a file is set to 'keep' it will be None unless # the data has been resubmitted or the original data # was passed in, it might be a bit cleaner to look # at the field.action to determine which files to # keep and which ones to delete/replace... if file_metas[idx] is None or file_metas[idx]['data'].startswith('@') } files_to_keep |= multi_files_to_keep # delete all files which are not part of the updated form # if no files are given, delete all files belonging to the submission trash = [f for f in submission.files if f.note not in files_to_keep] for f in trash: self.session.delete(f) if trash and inspect(submission).persistent: self.session.refresh(submission) # store the new files in the separate table for field_id in files_to_add: field = getattr(form, field_id) f = FormFile( # type:ignore[misc] id=random_token(), name=field.filename, note=field_id, reference=as_fileintent( content=field.file, filename=field.filename ) ) submission.files.append(f) # replace the data in the submission with a reference submission.data[field_id]['data'] = '@{}'.format(f.id) # we need to mark these changes as only top-level json changes # are automatically propagated submission.data.changed() # type:ignore[attr-defined] for field_id, indeces in multi_files.items(): datalist = [] # we can't use enumerate because we need to guard against # old_keys t new_idx = 0 for old_idx in indeces: data = submission.data[field_id][old_idx] old_key = f'{field_id}:{old_idx}' new_key = f'{field_id}:{new_idx}' if old_key in multi_files_to_keep: # update the key in the note field if the index changed if old_idx != new_idx: for f in submission.files: if f.note == old_key: f.note = new_key break else: field = getattr(form, field_id)[old_idx] if getattr(field, 'file', None) is None: # skip this subfield continue f = FormFile( # type:ignore[misc] id=random_token(), name=field.filename, note=new_key, reference=as_fileintent( content=field.file, filename=field.filename ) ) submission.files.append(f) # replace the data in the submission with a reference data['data'] = '@{}'.format(f.id) datalist.append(data) new_idx += 1 if submission.data[field_id] != datalist: submission.data[field_id] = datalist # we need to mark these changes as only top-level json changes # are automatically propagated submission.data.changed() # type:ignore[attr-defined]
[docs] def remove_old_pending_submissions( self, older_than: datetime, include_external: bool = False ) -> None: """ Removes all pending submissions older than the given date. The date is expected to be in UTC! """ if older_than.tzinfo is None: older_than = replace_timezone(older_than, 'UTC') submissions = self.query() # delete the ones that were never modified submissions = submissions.filter(FormSubmission.state == 'pending') submissions = submissions.filter( FormSubmission.last_change < older_than) if not include_external: submissions = submissions.filter(FormSubmission.name != None) for submission in submissions: self.session.delete(submission)
[docs] def by_state(self, state: SubmissionState) -> Query[FormSubmission]: return self.query().filter(FormSubmission.state == state)
# FIXME: Why are we returning a list here?
[docs] def by_name(self, name: str) -> list[FormSubmission]: """ Return all submissions for the given form-name. """ return self.query().filter(FormSubmission.name == name).all()
[docs] def by_id( self, id: UUID, state: SubmissionState | None = None, current_only: bool = False ) -> FormSubmission | None: """ Return the submission by id. :state: Only if the submission matches the given state. :current_only: Only if the submission is not older than one hour. """ query = self.query().filter(FormSubmission.id == id) if state is not None: query = query.filter(FormSubmission.state == state) if current_only: an_hour_ago = utcnow() - timedelta(hours=1) query = query.filter(FormSubmission.last_change >= an_hour_ago) return query.first()
[docs] def delete(self, submission: FormSubmission) -> None: """ Deletes the given submission and all the files belonging to it. """ self.session.delete(submission) self.session.flush()
[docs] class FormRegistrationWindowCollection( GenericCollection[FormRegistrationWindow] ): def __init__(self, session: Session, name: str | None = None): super().__init__(session)
[docs] self.name = name
@property
[docs] def model_class(self) -> type[FormRegistrationWindow]: return FormRegistrationWindow
[docs] def query(self) -> Query[FormRegistrationWindow]: query = super().query() if self.name: query = query.filter(FormRegistrationWindow.name == self.name) return query
[docs] class SurveyDefinitionCollection: """ Manages a collection of surveys. """ def __init__(self, session: Session):
[docs] self.session = session
[docs] def query(self) -> Query[SurveyDefinition]: return self.session.query(SurveyDefinition)
[docs] def add( self, title: str, definition: str, type: str = 'generic', meta: dict[str, Any] | None = None, content: dict[str, Any] | None = None, name: str | None = None, ) -> SurveyDefinition: """ Add the given survey to the database. """ # look up the right class depending on the type survey = SurveyDefinition.get_polymorphic_class( type, SurveyDefinition)() survey.name = name or normalize_for_url(title) survey.title = title survey.definition = definition survey.meta = meta or {} survey.content = content or {} # try to parse the survey (which will throw errors if there are # problems) assert survey.form_class self.session.add(survey) self.session.flush() return survey
[docs] def delete( self, name: str, with_submission_windows: bool = False, handle_submissions: SurveySubmissionHandler | None = None, handle_submission_windows: SubmissionWindowHandler | None = None, ) -> None: """ Delete the given form. Only possible if there are no submissions associated with it, or if ``with_submissions`` is True. Note that pending submissions are removed automatically, only complete submissions have a bearing on ``with_submissions``. Pass two callbacks to handle additional logic before deleting the objects. """ submissions = self.session.query(SurveySubmission) submissions = submissions.filter(SurveySubmission.name == name) if handle_submissions: handle_submissions(submissions) submissions.delete() if with_submission_windows: submission_windows = self.session.query(SurveySubmissionWindow) submission_windows = submission_windows.filter_by(name=name) if handle_submission_windows: handle_submission_windows(submission_windows) submission_windows.delete() self.session.flush() # this will fail if there are any submissions left self.query().filter(SurveyDefinition.name == name).delete('fetch') self.session.flush()
[docs] def by_name(self, name: str) -> SurveyDefinition | None: """ Returns the given form by name or None. """ return self.query().filter(SurveyDefinition.name == name).first()
[docs] class SurveySubmissionCollection: """ Manages a collection of survey submissions. """ def __init__(self, session: Session, name: str | None = None):
[docs] self.session = session
[docs] self.name = name
[docs] def query(self) -> Query[SurveySubmission]: query = self.session.query(SurveySubmission) if self.name is not None: query = query.filter(SurveySubmission.name == self.name) return query
[docs] def add( self, name: str | None, form: Form, submission_window: SurveySubmissionWindow | None = None, id: UUID | None = None, meta: dict[str, Any] | None = None, ) -> SurveySubmission: """ Takes a filled-out survey instance and stores the submission in the database. The survey instance is expected to have a ``_source`` parameter, which contains the source used to build the szrvey (as only surveys with this source may be stored). This method expects the name of the survey definition stored in the database. Use :meth:`add_external` to add a submissions whose definition is not stored in the form_definitions table. """ assert hasattr(form, '_source') form.validate() if name is None: definition = None else: definition = ( self.session.query(SurveyDefinition) .filter_by(name=name).one()) if definition is None: submission_window = None else: submission_window = submission_window submission = SurveySubmission( id=id or uuid4(), name=name, meta=meta or {}, submission_window=submission_window ) # extensions are inherited from definitions if definition: assert not submission.extensions, """ For submissions based on definitions, the extensions need to be defined on the definition! """ submission.extensions = definition.extensions self.update(submission, form) self.session.add(submission) self.session.flush() return submission
[docs] def update( self, submission: SurveySubmission, form: Form, exclude: Collection[str] | None = None ) -> None: """ Takes a submission and a survey and updates the submission data as well as the files stored in a separate table. """ assert submission.id # ignore certain fields exclude = set(exclude) if exclude else set() exclude.add(form.meta.csrf_field_name) # never include the csrf token assert hasattr(form, '_source') submission.definition = form._source submission.data = { k: v for k, v in form.data.items() if k not in exclude } submission.update_title(form)
[docs] def by_name(self, name: str) -> list[SurveySubmission]: """ Return all submissions for the given form-name. """ return self.query().filter(SurveySubmission.name == name).all()
[docs] def by_id( self, id: UUID, current_only: bool = False ) -> SurveySubmission | None: """ Return the submission by id. :state: Only if the submission matches the given state. :current_only: Only if the submission is not older than one hour. """ query = self.query().filter(SurveySubmission.id == id) if current_only: an_hour_ago = utcnow() - timedelta(hours=1) query = query.filter(SurveySubmission.last_change >= an_hour_ago) return query.first()
[docs] def delete(self, submission: SurveySubmission) -> None: """ Deletes the given submission and all the files belonging to it. """ self.session.delete(submission) self.session.flush()
[docs] class SurveySubmissionWindowCollection( GenericCollection[SurveySubmissionWindow] ): def __init__(self, session: Session, name: str | None = None): super().__init__(session)
[docs] self.name = name
@property
[docs] def model_class(self) -> type[SurveySubmissionWindow]: return SurveySubmissionWindow
[docs] def query(self) -> Query[SurveySubmissionWindow]: query = super().query() if self.name: query = query.filter(SurveySubmissionWindow.name == self.name) return query
[docs] class SurveyCollection: """ Manages a collection of surveys and survey-submissions. """ def __init__(self, session: Session):
[docs] self.session = session
@property
[docs] def definitions(self) -> SurveyDefinitionCollection: return SurveyDefinitionCollection(self.session)
@property
[docs] def submissions(self) -> SurveySubmissionCollection: return SurveySubmissionCollection(self.session)
@property
[docs] def submission_windows(self) -> SurveySubmissionWindowCollection: return SurveySubmissionWindowCollection(self.session)
@overload
[docs] def scoped_submissions( self, name: str, ensure_existance: Literal[False] ) -> SurveySubmissionCollection: ...
@overload def scoped_submissions( self, name: str, ensure_existance: bool = True ) -> SurveySubmissionCollection | None: ... def scoped_submissions( self, name: str, ensure_existance: bool = True ) -> SurveySubmissionCollection | None: if not ensure_existance or self.definitions.by_name(name): return SurveySubmissionCollection(self.session, name) return None # FIXME: This should use Intersection[HasSubmissionsCount] since we # add a temporary attribute to the returned form definitions. # But we have to wait until this feature is available
[docs] def get_definitions_with_submission_count( self ) -> Iterator[FormDefinition]: """ Returns all form definitions and the number of submissions belonging to those definitions, in a single query. The number of submissions is stored on the form definition under the ``submissions_count`` attribute. Only submissions which are 'complete' are considered. """ submissions_ = self.session.query( FormSubmission.name, func.count(FormSubmission.id).label('count') ) submissions = submissions_.group_by(FormSubmission.name).subquery() definitions = self.session.query(FormDefinition, submissions.c.count) definitions = definitions.outerjoin( submissions, submissions.c.name == FormDefinition.name ) definitions = definitions.order_by(FormDefinition.name) for survey, submissions_count in definitions.all(): survey.submissions_count = submissions_count or 0 yield survey