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
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):
@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]
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]
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)
@property
[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]
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]
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)
@property
[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):
@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