from __future__ import annotations
from onegov.activity import ActivityFilter
from onegov.activity import Attendee, AttendeeCollection
from onegov.activity import Booking, BookingCollection
from onegov.activity import InvoiceCollection, InvoiceItem
from onegov.activity import Occasion, OccasionCollection, OccasionNeed
from onegov.activity import Period, PeriodCollection
from onegov.activity import Volunteer, VolunteerCollection
from onegov.activity.utils import is_valid_group_code
from onegov.feriennet import FeriennetApp
from onegov.feriennet.collections import BillingCollection
from onegov.feriennet.collections import MatchCollection
from onegov.feriennet.collections import NotificationTemplateCollection
from onegov.feriennet.collections import OccasionAttendeeCollection
from onegov.feriennet.collections import VacationActivityCollection
from onegov.feriennet.collections.match import OccasionState
from onegov.feriennet.models import Calendar
from onegov.feriennet.models import GroupInvite
from onegov.feriennet.models import InvoiceAction, VacationActivity
from onegov.feriennet.models import NotificationTemplate
from onegov.feriennet.models import VolunteerCart
from onegov.feriennet.models import VolunteerCartAction
from onegov.org.converters import keywords_converter
from onegov.core.converters import integer_range_converter, LiteralConverter
from uuid import UUID
from typing import Literal, TYPE_CHECKING
if TYPE_CHECKING:
from onegov.activity.models import PeriodMeta
from onegov.feriennet.request import FeriennetRequest
@FeriennetApp.path(
model=VacationActivityCollection,
path='/activities',
converters={
'filter': keywords_converter,
'pages': integer_range_converter
})
[docs]
def get_vacation_activities(
request: FeriennetRequest,
app: FeriennetApp,
pages: tuple[int, int] | None = None,
filter: dict[str, list[str]] | None = None
) -> VacationActivityCollection:
filter_obj = ActivityFilter(**filter) if filter else ActivityFilter()
if not request.is_organiser:
filter_obj.period_ids = (
{app.active_period.id} if app.active_period else set())
collection = VacationActivityCollection(
session=app.session(),
pages=pages,
identity=request.identity,
filter=filter_obj
)
# prevent large lookups
return collection.by_page_range(
collection.limit_range(collection.pages, 'down')
)
@FeriennetApp.path(
model=VacationActivity,
path='/activity/{name}')
[docs]
def get_vacation_activity(
request: FeriennetRequest,
name: str
) -> VacationActivity | None:
return VacationActivityCollection(
request.session, identity=request.identity).by_name(name)
@FeriennetApp.path(
model=Occasion,
path='/occasions/{id}',
converters={'id': UUID})
[docs]
def get_occasion(request: FeriennetRequest, id: UUID) -> Occasion | None:
return OccasionCollection(request.session).by_id(id)
@FeriennetApp.path(
model=PeriodCollection,
path='/periods')
[docs]
def get_periods(request: FeriennetRequest) -> PeriodCollection:
return PeriodCollection(request.session)
@FeriennetApp.path(
model=Period,
path='/period/{id}',
converters={'id': UUID})
[docs]
def get_period(request: FeriennetRequest, id: UUID) -> Period | None:
return PeriodCollection(request.session).by_id(id)
@FeriennetApp.path(
model=BookingCollection,
path='/my-bookings',
converters={'period_id': UUID})
[docs]
def get_my_bookings(
request: FeriennetRequest,
app: FeriennetApp,
period_id: UUID | None = None,
username: str | None = None
) -> BookingCollection:
# only admins can actually specify the username
if not request.is_admin:
username = request.current_username
# the default username is the current user
if not username:
username = request.current_username
# the default period is the active period or the first we can find
if not period_id:
period = app.default_period
if period:
period_id = period.id
return BookingCollection(app.session(), period_id, username)
@FeriennetApp.path(
model=Booking,
path='/booking/{id}',
converters={'id': UUID})
[docs]
def get_booking(request: FeriennetRequest, id: UUID) -> Booking | None:
return BookingCollection(request.session).by_id(id)
@FeriennetApp.path(
model=Attendee,
path='/attendee/{id}',
converters={'id': UUID})
[docs]
def get_attendee(request: FeriennetRequest, id: UUID) -> Attendee | None:
return AttendeeCollection(request.session).by_id(id)
@FeriennetApp.path(
model=MatchCollection,
path='/matching',
converters={
'period_id': UUID,
'states': [LiteralConverter(OccasionState)]
})
[docs]
def get_matches(
request: FeriennetRequest,
app: FeriennetApp,
period_id: UUID | None,
states: list[OccasionState] | None = None
) -> MatchCollection | None:
# the default period is the active period or the first we can find
period: Period | PeriodMeta | None
if not period_id:
period = app.default_period
else:
period = PeriodCollection(app.session()).by_id(period_id)
if not period:
return None
return MatchCollection(app.session(), period, states)
@FeriennetApp.path(
model=BillingCollection,
path='/billing',
converters={
'period_id': UUID,
'expand': bool,
'payment_state': LiteralConverter('paid', 'unpaid')
}
)
[docs]
def get_billing(
request: FeriennetRequest,
app: FeriennetApp,
period_id: UUID,
username: str | None = None,
expand: bool = False,
state: Literal['paid', 'unpaid'] = 'unpaid'
) -> BillingCollection | None:
# the default period is the active period or the first we can find
period: Period | PeriodMeta | None
if not period_id:
period = app.default_period
else:
period = PeriodCollection(app.session()).by_id(period_id)
if not period:
return None
return BillingCollection(request, period, username, expand, state)
@FeriennetApp.path(
model=InvoiceAction,
path='/invoice-action/{id}/{action}',
converters={
'id': UUID,
'action': LiteralConverter(
'mark-paid',
'mark-unpaid',
'remove-manual'
),
'extend_to': LiteralConverter('invoice', 'family')
})
[docs]
def get_invoice_action(
request: InvoiceAction,
app: FeriennetApp,
id: UUID,
action: Literal['mark-paid', 'mark-unpaid', 'remove-manual'],
extend_to: Literal['invoice', 'family'] | None = None
) -> InvoiceAction | None:
action_obj = InvoiceAction(
session=app.session(),
id=id,
action=action,
extend_to=extend_to
)
return action_obj if action_obj.valid else None
@FeriennetApp.path(
model=InvoiceCollection,
path='/my-bills',
converters={'invoice': UUID})
[docs]
def get_my_invoices(
request: FeriennetRequest,
app: FeriennetApp,
username: str | None = None,
invoice: UUID | None = None
) -> InvoiceCollection | None:
# only admins can actually specify the username/invoice
if not request.is_admin:
username, invoice = None, None
# the default username is the current user
if not username:
username = request.current_username
# the default period is the active period
if not invoice and request.app.default_period:
invoice = request.app.default_period.id
user_id = request.app.user_ids_by_name.get(username)
# the invoice has to exist
if not invoice:
return None
# XXX username should be user_id, invoice should be period_id
# this should be changed, but needs to be changed by replacing
# the username everywhere
return app.invoice_collection(period_id=invoice, user_id=user_id)
@FeriennetApp.path(
model=InvoiceItem,
path='/invoice-item/{id}',
converters={'id': UUID})
[docs]
def get_my_invoice_item(
request: FeriennetRequest,
id: UUID
) -> InvoiceItem | None:
return request.session.query(InvoiceItem).filter_by(id=id).first()
@FeriennetApp.path(
model=OccasionAttendeeCollection,
path='/attendees/{activity_name}',
converters={'period_id': UUID})
[docs]
def get_occasion_attendee_collection(
request: FeriennetRequest,
app: FeriennetApp,
activity_name: str,
period_id: UUID | None = None
) -> OccasionAttendeeCollection | None:
# load the activity
activity = get_vacation_activity(request, activity_name)
if not activity:
return None
# the default period is the active period or the first we can find
period: Period | PeriodMeta | None
if not period_id:
period = app.default_period
else:
period = PeriodCollection(app.session()).by_id(period_id)
if not period:
return None
# non-admins are automatically limited to the activites they own
if request.is_admin:
username = None
else:
username = request.current_username
return OccasionAttendeeCollection(
app.session(), period, activity, username)
@FeriennetApp.path(
model=NotificationTemplateCollection,
path='/notifications')
[docs]
def get_notification_template_collection(
app: FeriennetApp
) -> NotificationTemplateCollection:
return NotificationTemplateCollection(app.session())
@FeriennetApp.path(
model=NotificationTemplate,
path='/notification/{id}',
converters={'id': UUID, 'period_id': UUID})
[docs]
def get_notification_template(
app: FeriennetApp,
id: UUID,
period_id: UUID | None
) -> NotificationTemplate | None:
template = NotificationTemplateCollection(app.session()).by_id(id)
if not period_id and not app.active_period:
return None
if template:
if not period_id:
assert app.active_period is not None
period_id = app.active_period.id
template.period_id = period_id
return template
@FeriennetApp.path(
model=Calendar,
path='/calendar/{name}/{token}')
[docs]
def get_calendar(
request: FeriennetRequest,
name: str,
token: str
) -> Calendar | None:
return Calendar.from_name_and_token(request.session, name, token)
@FeriennetApp.path(
model=GroupInvite,
path='/join/{group_code}')
[docs]
def get_group_invite(
request: FeriennetRequest,
group_code: str,
username: str | None = None
) -> GroupInvite | None:
group_code = group_code.upper()
if not is_valid_group_code(group_code):
return None
if not request.is_admin or not username:
username = request.current_username
invite = GroupInvite(request.session, group_code, username)
return invite.exists and invite or None
@FeriennetApp.path(
model=OccasionNeed,
path='/occasion-need/{id}',
converters={'id': UUID})
[docs]
def get_occasion_need(
request: FeriennetRequest,
id: UUID
) -> OccasionNeed | None:
return request.session.query(OccasionNeed).filter_by(id=id).first()
@FeriennetApp.path(
model=VolunteerCart,
path='/volunteer-cart')
[docs]
def get_volunteer_cart(request: FeriennetRequest) -> VolunteerCart:
return VolunteerCart.from_request(request)
@FeriennetApp.path(
model=VolunteerCartAction,
path='/volunteer-cart-action/{action}/{target}',
converters={'action': LiteralConverter('add', 'remove'), 'target': UUID})
[docs]
def get_volunteer_cart_action(
request: FeriennetRequest,
action: Literal['add', 'remove'],
target: UUID
) -> VolunteerCartAction:
return VolunteerCartAction(action, target)
@FeriennetApp.path(
model=VolunteerCollection,
path='/volunteers/{period_id}',
converters={'period_id': UUID})
[docs]
def get_volunteers(
request: FeriennetRequest,
period_id: UUID
) -> VolunteerCollection | None:
if not period_id:
return None
if not request.app.show_volunteers(request):
return None
period = request.app.periods_by_id.get(period_id.hex)
if not period:
return None
return VolunteerCollection(request.session, period)
@FeriennetApp.path(
model=Volunteer,
path='/volunteer/{id}',
converters={'id': UUID})
[docs]
def get_volunteer(request: FeriennetRequest, id: UUID) -> Volunteer | None:
return VolunteerCollection(request.session, None).by_id(id)