Source code for activity.matching.score

from __future__ import annotations

import hashlib

from decimal import Decimal
from onegov.activity.models import Activity, Attendee, Booking, Occasion
from onegov.user import User
from sqlalchemy import func


from typing import Any, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Callable
    from sqlalchemy.orm import Session
    from typing import Self


[docs] class Scoring: """ Provides scoring based on a number of criteria. A criteria is a callable which takes a booking and returns a score. The final score is the sum of all criteria scores. """
[docs] criteria: list[Callable[[Booking], float]]
def __init__( self, criteria: list[Callable[[Booking], float]] | None = None ) -> None: self.criteria = criteria or [PreferMotivated()]
[docs] def __call__(self, booking: Booking) -> Decimal: return Decimal(sum(criterium(booking) for criterium in self.criteria))
@classmethod
[docs] def from_settings( cls, settings: dict[str, Any], session: Session ) -> Self: scoring = cls() # always prefer groups scoring.criteria.append(PreferGroups.from_session(session)) if settings.get('prefer_in_age_bracket'): scoring.criteria.append( PreferInAgeBracket.from_session(session)) if settings.get('prefer_organiser'): scoring.criteria.append( PreferOrganiserChildren.from_session(session)) if settings.get('prefer_admins'): scoring.criteria.append( PreferAdminChildren.from_session(session)) return scoring
@property
[docs] def settings(self) -> dict[str, Any]: classes = {c.__class__ for c in self.criteria} settings = {} if PreferInAgeBracket in classes: settings['prefer_in_age_bracket'] = True if PreferOrganiserChildren in classes: settings['prefer_organiser'] = True if PreferAdminChildren in classes: settings['prefer_admins'] = True return settings
[docs] class PreferMotivated: """ Scores "motivated" bookings higher. A motivated booking is simply a booking with a higher priority (an attendee would favor a booking he's excited about.) """ @classmethod
[docs] def from_session(cls, session: Session) -> Self: return cls()
[docs] def __call__(self, booking: Booking) -> int: return booking.priority
[docs] class PreferInAgeBracket: """ Scores bookings whose attendees fall into the age-bracket of the occasion higher. If the attendee falls into the age-bracket, the score is 1.0. Each year difference results in a penalty of 0.1, until 0.0 is reached. """ def __init__( self, get_age_range: Callable[[Booking], tuple[int, int]], get_attendee_age: Callable[[Booking], int] ):
[docs] self.get_age_range = get_age_range
[docs] self.get_attendee_age = get_attendee_age
@classmethod
[docs] def from_session(cls, session: Session) -> Self: attendees = None occasions = None def get_age_range(booking: Booking) -> tuple[int, int]: nonlocal occasions, session if occasions is None: occasions = { o.id: o.age for o in session.query(Occasion.id, Occasion.age) .filter(Occasion.period_id == booking.period_id)} return ( occasions[booking.occasion_id].lower, occasions[booking.occasion_id].upper - 1 ) def get_attendee_age(booking: Booking) -> int: nonlocal attendees, session if attendees is None: attendees = {a.id: a.age for a in session.query( Attendee.id, Attendee.age)} return attendees[booking.attendee_id] return cls(get_age_range, get_attendee_age)
[docs] def __call__(self, booking: Booking) -> float: min_age, max_age = self.get_age_range(booking) attendee_age = self.get_attendee_age(booking) if min_age <= attendee_age and attendee_age <= max_age: return 1.0 else: difference = min( abs(min_age - attendee_age), abs(max_age - attendee_age) ) return 1.0 - min(1.0, float(difference) / 10.0)
[docs] class PreferOrganiserChildren: """ Scores bookings of children higher if their parents are organisers. This is basically an incentive to become an organiser. A child whose parent is an organiser gets a score of 1.0, if the parent is not an organiser a score 0.0 is returned. """ def __init__(self, get_is_organiser_child: Callable[[Booking], bool]):
[docs] self.get_is_organiser_child = get_is_organiser_child
@classmethod
[docs] def from_session(cls, session: Session) -> Self: organisers = None def get_is_organiser_child(booking: Booking) -> bool: nonlocal organisers if organisers is None: organisers = { username for username, in session.query(Activity.username) .filter(Activity.id.in_( session.query(Occasion.activity_id) .filter(Occasion.period_id == booking.period_id) .subquery() )) } return booking.username in organisers return cls(get_is_organiser_child)
[docs] def __call__(self, booking: Booking) -> float: return 1.0 if self.get_is_organiser_child(booking) else 0.0
[docs] class PreferAdminChildren: """ Scores bookings of children higher if their parents are admins. """ def __init__(self, get_is_association_child: Callable[[Booking], bool]):
[docs] self.get_is_association_child = get_is_association_child
@classmethod
[docs] def from_session(cls, session: Session) -> Self: members = None def get_is_association_child(booking: Booking) -> bool: nonlocal members if members is None: members = { u.username for u in session.query(User) .filter(User.role == 'admin') .filter(User.active == True) } return booking.username in members return cls(get_is_association_child)
[docs] def __call__(self, booking: Booking) -> float: return self.get_is_association_child(booking) and 1.0 or 0.0
[docs] class PreferGroups: """ Scores group bookings higher than other bookings. Groups get a boost by size: - 2 people: 1.0 - 3 people: 0.8 - 4 people: 0.6 - more people: 0.5 This preference gives an extra boost to unprioritised bookings, to somewhat level out bookings in groups that used no star (otherwise a group might be split up because someone didn't star the booking). Additionally a unique boost between 0.010000000 to 0.099999999 is given to each group depending on the group name. This should ensure that competing groups generally do not have the same score. So an occasion will generally prefer the members of one group over members of another group. """ def __init__(self, get_group_score: Callable[[Booking], float]):
[docs] self.get_group_score = get_group_score
@classmethod
[docs] def from_session(cls, session: Session) -> Self: group_scores = None def unique_score_modifier(group_code: str) -> float: digest = hashlib.new( 'sha1', group_code.encode('utf-8'), usedforsecurity=False ).hexdigest()[:8] number = int(digest, 16) return float('0.0' + str(number)[:8]) def get_group_score(booking: Booking) -> float: nonlocal group_scores if group_scores is None: query = session.query(Booking).with_entities( Booking.group_code, func.count(Booking.group_code).label('count') ).filter( Booking.group_code != None, Booking.period_id == booking.period_id ).group_by( Booking.group_code ).having( func.count(Booking.group_code) > 1 ) group_scores = { r.group_code: max(.5, 1.0 - 0.2 * (r.count - 2)) + unique_score_modifier(r.group_code) for r in query } return group_scores.get(booking.group_code, 0) return cls(get_group_score)
[docs] def __call__(self, booking: Booking) -> float: return self.get_group_score(booking)