Source code for form.models.registration_window
from __future__ import annotations
import sedate
from onegov.core.orm import Base
from onegov.core.orm.mixins import TimestampMixin
from onegov.core.orm.types import UUID
from onegov.form.models.submission import FormSubmission
from sqlalchemy import and_
from sqlalchemy import Boolean
from sqlalchemy import Column
from sqlalchemy import Date
from sqlalchemy import ForeignKey
from sqlalchemy import Integer
from sqlalchemy import or_
from sqlalchemy import Text
from sqlalchemy import text
from sqlalchemy.dialects.postgresql import ExcludeConstraint
from sqlalchemy.orm import object_session, relationship
from sqlalchemy.schema import CheckConstraint
from sqlalchemy.sql.elements import quoted_name
from uuid import uuid4
from typing import TYPE_CHECKING
if TYPE_CHECKING:
import uuid
from datetime import date, datetime
from onegov.form.models.definition import FormDefinition
[docs]
daterange = Column( # type:ignore[call-overload]
quoted_name('DATERANGE("start", "end")', quote=False))
[docs]
class FormRegistrationWindow(Base, TimestampMixin):
""" Defines a registration window during which a form definition
may be used to create submissions.
Submissions created thusly are attached to the currently active
registration window.
Registration windows may not overlap.
"""
#: the public id of the registraiton window
[docs]
id: Column[uuid.UUID] = Column(
UUID, # type:ignore[arg-type]
primary_key=True,
default=uuid4
)
#: the name of the form to which this registration window belongs
#: the form to which this registration window belongs
[docs]
form: relationship[FormDefinition] = relationship(
'FormDefinition',
back_populates='registration_windows'
)
#: true if the registration window is enabled
#: the start date of the window
#: the end date of the window
#: the timezone of the window
#: the number of spots (None => unlimited)
#: enable an overflow of submissions
#: submissions linked to this registration window
[docs]
submissions: relationship[list[FormSubmission]] = relationship(
FormSubmission,
back_populates='registration_window'
)
[docs]
__table_args__ = (
# ensures that there are no overlapping date ranges within one form
ExcludeConstraint(
(name, '='), (daterange, '&&'),
name='no_overlapping_registration_windows',
using='gist'
),
# ensures that there are no adjacent date ranges
# (end on the same day as next start)
ExcludeConstraint(
(name, '='), (daterange, '-|-'),
name='no_adjacent_registration_windows',
using='gist'
),
# ensures that start <= end
CheckConstraint(
'"start" <= "end"',
name='start_smaller_than_end'
),
)
@property
[docs]
def localized_start(self) -> datetime:
return sedate.align_date_to_day(
sedate.standardize_date(
sedate.as_datetime(self.start), self.timezone
), self.timezone, 'down'
)
@property
[docs]
def localized_end(self) -> datetime:
return sedate.align_date_to_day(
sedate.standardize_date(
sedate.as_datetime(self.end), self.timezone
), self.timezone, 'up'
)
[docs]
def disassociate(self) -> None:
""" Disassociates all records linked to this window. """
for submission in self.submissions:
submission.disclaim()
submission.registration_window_id = None
@property
@property
@property
[docs]
def in_the_present(self) -> bool:
return self.localized_start <= sedate.utcnow() <= self.localized_end
[docs]
def accepts_submissions(self, required_spots: int = 1) -> bool:
assert required_spots > 0
if not self.enabled:
return False
if not self.in_the_present:
return False
if self.overflow:
return True
if self.limit is None:
return True
return self.available_spots >= required_spots
@property
[docs]
def next_submission(self) -> FormSubmission | None:
""" Returns the submission next in line. In other words, the next
submission in order of first come, first serve.
"""
q = object_session(self).query(FormSubmission)
q = q.filter(FormSubmission.registration_window_id == self.id)
q = q.filter(FormSubmission.state == 'complete')
q = q.filter(or_(
FormSubmission.claimed == None,
and_(
FormSubmission.claimed > 0,
FormSubmission.claimed < FormSubmission.spots,
)
))
q = q.order_by(FormSubmission.created)
return q.first()
@property
[docs]
def available_spots(self) -> int:
assert self.limit is not None
return max(self.limit - self.claimed_spots - self.requested_spots, 0)
@property
[docs]
def claimed_spots(self) -> int:
""" Returns the number of actually claimed spots. """
return object_session(self).execute(text("""
SELECT SUM(COALESCE(claimed, 0))
FROM submissions
WHERE registration_window_id = :id
AND submissions.state = 'complete'
"""), {'id': self.id}).scalar() or 0
@property
[docs]
def requested_spots(self) -> int:
""" Returns the number of requested spots.
When the claim has not been made yet, `spots` are counted as
requested. When the claim has been partially made, the difference is
counted as requested. If the claim has been fully made, the result is
0. If the claim has been relinquished, the result is 0.
"""
return object_session(self).execute(text("""
SELECT GREATEST(
SUM(
CASE WHEN claimed IS NULL THEN spots
WHEN claimed = 0 THEN 0
ELSE spots - claimed
END
), 0
)
FROM submissions
WHERE registration_window_id = :id
AND submissions.state = 'complete'
"""), {'id': self.id}).scalar() or 0