org.utils ========= .. py:module:: org.utils Attributes ---------- .. autoapisummary:: org.utils._T org.utils.EMPTY_PARAGRAPHS org.utils.HASHTAG org.utils.IMG_URLS org.utils.libres_error_messages org.utils.ORDERED_ACCESS Classes ------- .. autoapisummary:: org.utils.ReservationInfo org.utils.AllocationEventInfo org.utils.FindYourSpotEventInfo Functions --------- .. autoapisummary:: org.utils.djb2_hash org.utils.get_random_color org.utils.get_user_color org.utils.get_extension_color org.utils.add_class_to_node org.utils.annotate_html org.utils.remove_empty_paragraphs org.utils.set_image_sizes org.utils.parse_fullcalendar_request org.utils.render_time_range org.utils.get_libres_error org.utils.show_libres_error org.utils.predict_next_daterange org.utils.predict_next_value org.utils.group_by_column org.utils.keywords_first org.utils.hashtag_elements org.utils.ticket_directory_groups org.utils.user_group_emails_for_new_ticket org.utils.widest_access org.utils.extract_categories_and_subcategories org.utils.format_phone_number Module Contents --------------- .. py:data:: _T .. py:data:: EMPTY_PARAGRAPHS .. py:data:: HASHTAG .. py:data:: IMG_URLS .. py:function:: djb2_hash(text: str, size: int) -> int Implementation of the djb2 hash, a simple hash function with a configurable table size. ** Do NOT use for cryptography! ** .. py:function:: get_random_color(seed: str, lightness: float, saturation: float) -> str Gets a random color using the given seed (a text value). Since the colorspace is very limited there are lots of collisions. .. py:function:: get_user_color(username: str) -> str Gets a user color for each username which is used for the user-initials-* elements. Each username is mapped to a color. :returns: The user color in an css rgb string. .. py:function:: get_extension_color(extension: str) -> str Gets an extension color for each file extension. This is similar to :func:`get_user_color`, but returns a darker color (text is white). .. py:function:: add_class_to_node(node: lxml.etree._Element, classname: str) -> None Adds the given classname to the given lxml node's class list. .. py:function:: annotate_html(html: markupsafe.Markup, request: onegov.core.request.CoreRequest | None = None) -> markupsafe.Markup annotate_html(html: None, request: onegov.core.request.CoreRequest | None = None) -> None Takes the given html and annotates the following elements for some advanced styling: * Every paragraph containing an img element will be marked with the `has-img` class. * If a link is found which points to a youtube or a vimeo video, the link itself as well as the surrounding paragraph is marked with the `has-video` class * If a hashtag is found, the paragraph gets the 'has-hashtag' class. .. py:function:: remove_empty_paragraphs(html: None) -> None remove_empty_paragraphs(html: markupsafe.Markup) -> markupsafe.Markup .. py:function:: set_image_sizes(images: list[lxml.etree._Element], request: onegov.core.request.CoreRequest) -> None .. py:function:: parse_fullcalendar_request(request: onegov.core.request.CoreRequest, timezone: str) -> tuple[datetime.datetime, datetime.datetime] | tuple[None, None] Parses start and end from the given fullcalendar request. It is expected that no timezone is passed (the default). See ``_ :returns: A tuple of timezone-aware datetime objects or (None, None). .. py:function:: render_time_range(start: datetime.datetime | datetime.time, end: datetime.datetime | datetime.time) -> str .. py:class:: ReservationInfo(resource: onegov.reservation.Resource, reservation: onegov.reservation.Reservation, request: onegov.org.request.OrgRequest) .. py:attribute:: __slots__ :value: ('resource', 'reservation', 'request', 'translate') .. py:attribute:: resource .. py:attribute:: reservation .. py:attribute:: request .. py:attribute:: translate .. py:property:: date :type: str .. py:property:: warning :type: str | None .. py:property:: time :type: str .. py:property:: delete_link :type: str .. py:property:: price :type: onegov.pay.types.PriceDict | None .. py:method:: as_dict() -> dict[str, Any] .. py:class:: AllocationEventInfo(resource: onegov.reservation.Resource, allocation: onegov.reservation.Allocation, availability: float, request: onegov.org.request.OrgRequest) .. py:attribute:: __slots__ :value: ('resource', 'allocation', 'availability', 'request', 'translate') .. py:attribute:: resource .. py:attribute:: allocation .. py:attribute:: availability .. py:attribute:: request .. py:attribute:: translate .. py:method:: from_allocations(request: onegov.org.request.OrgRequest, resource: onegov.reservation.Resource, allocations: collections.abc.Iterable[onegov.reservation.Allocation]) -> list[Self] :classmethod: .. py:property:: event_start :type: str .. py:property:: event_end :type: str .. py:property:: event_identification :type: str .. py:property:: event_time :type: str .. py:property:: quota :type: int .. py:property:: quota_left :type: int .. py:property:: event_title :type: str .. py:property:: event_classes :type: collections.abc.Iterator[str] .. py:property:: occupancy_link :type: str .. py:property:: event_actions :type: collections.abc.Iterator[onegov.org.elements.Link] .. py:method:: as_dict() -> dict[str, Any] .. py:class:: FindYourSpotEventInfo(allocation: onegov.reservation.Allocation, slot_time: DateRange | None, availability: float, quota_left: int, request: onegov.org.request.OrgRequest, *, adjustable: bool = False) .. py:attribute:: __slots__ :value: ('allocation', 'slot_time', 'availability', 'quota_left', 'request', 'translate', 'adjustable') .. py:attribute:: allocation .. py:attribute:: slot_time .. py:attribute:: availability .. py:attribute:: quota_left .. py:attribute:: request .. py:attribute:: adjustable :value: False .. py:attribute:: translate .. py:property:: event_start :type: str .. py:property:: event_end :type: str .. py:property:: event_time :type: str .. py:property:: quota :type: int .. py:property:: whole_day :type: str .. py:property:: partly_available :type: str .. py:property:: available :type: str .. py:property:: event_classes :type: collections.abc.Iterator[str] .. py:property:: css_class :type: str .. py:property:: reserveurl :type: str .. py:data:: libres_error_messages .. py:function:: get_libres_error(e: Exception, request: onegov.org.request.OrgRequest) -> str .. py:function:: show_libres_error(e: Exception, request: onegov.org.request.OrgRequest) -> None Shows a human readable error message for the given libres exception, using request.alert. .. py:function:: predict_next_daterange(dateranges: collections.abc.Sequence[DateRange], min_probability: float = 0.8, tzinfo: TzInfo | None = None) -> DateRange | None Takes a list of dateranges (start, end) and tries to predict the next daterange in the list. See :func:`predict_next_value` for more information. .. py:function:: predict_next_value(values: collections.abc.Sequence[_T], min_probability: float = 0.8) -> _T | None predict_next_value(values: collections.abc.Sequence[_T], min_probability: float, compute_delta: collections.abc.Callable[[_T, _T], _DeltaT], add_delta: collections.abc.Callable[[_T, _DeltaT], _T | None]) -> _T | None predict_next_value(values: collections.abc.Sequence[_T], min_probability: float = 0.8, *, compute_delta: collections.abc.Callable[[_T, _T], _DeltaT], add_delta: collections.abc.Callable[[_T, _DeltaT], _T | None]) -> _T | None Takes a list of values and tries to predict the next value in the series. Meant to work on a small set of ranges (with first predictions appearing with only three values), this algorithm will look at all possible deltas between the values and keep track of the probability of delta y following delta x. If the delta between the second last and last value has a high probability of being followed by some delta p, then delta p is used to predict the next range. If the probability is too low (signified by min_probability), then None is returned. For large ranges better statistical models should be used. Here we are concerned with small series of data to answer the question "if a user selected three values, what will his fourth be?" If we for example know that the user selected 1, 2 and 3, then 4 is the next probable value in the series. .. py:function:: group_by_column(request: onegov.org.request.OrgRequest, query: sqlalchemy.orm.Query[_T], group_column: sqlalchemy.Column[str] | sqlalchemy.Column[str | None], sort_column: sqlalchemy.Column[_SortT], default_group: str | None = None, transform: collections.abc.Callable[[_T], _T] | None = None) -> dict[str, list[_T]] group_by_column(request: onegov.org.request.OrgRequest, query: sqlalchemy.orm.Query[_T], group_column: sqlalchemy.Column[str] | sqlalchemy.Column[str | None], sort_column: sqlalchemy.Column[_SortT], default_group: str | None, transform: collections.abc.Callable[[_T], _TransformedT]) -> dict[str, list[_TransformedT]] group_by_column(request: onegov.org.request.OrgRequest, query: sqlalchemy.orm.Query[_T], group_column: sqlalchemy.Column[str] | sqlalchemy.Column[str | None], sort_column: sqlalchemy.Column[_SortT], default_group: str | None = None, *, transform: collections.abc.Callable[[_T], _TransformedT]) -> dict[str, list[_TransformedT]] Groups the given query by the given group. :param request: The current request used for translation and to exclude invisible records. :param query: The query that should be grouped :param group_column: The column by which the grouping should happen. :param sort_column: The column by which the records should be sorted. :param default_group: The group in use if the found group is empty (optional). :param transform: Called with each record to transform the result (optional). .. py:function:: keywords_first(keywords: collections.abc.Sequence[str]) -> collections.abc.Callable[[str], tuple[int, str]] Returns a sort key which prefers values matching the given keywords before other values which are sorted alphabetically. .. py:function:: hashtag_elements(request: onegov.org.request.OrgRequest, text: str) -> markupsafe.Markup Takes a text and adds html around the hashtags found inside. .. py:function:: ticket_directory_groups(session: sqlalchemy.orm.Session) -> collections.abc.Iterator[str] Yields all ticket groups. For example: ('Sportanbieter', 'Verein') If no groups exist, returns an empty generator. .. py:function:: user_group_emails_for_new_ticket(request: onegov.core.request.CoreRequest, ticket: onegov.ticket.Ticket) -> set[str] The user can be part of a UserGroup that defines directories. This means the users in this group are interested in a subset of tickets. The group is determined by the Ticket group. This allows for more granular control over who gets notified. .. py:data:: ORDERED_ACCESS :value: ('private', 'member', 'secret_mtan', 'mtan', 'secret', 'public') .. py:function:: widest_access(*accesses: str) -> str .. py:function:: extract_categories_and_subcategories(categories: list[dict[str, list[str]] | str], flattened: Literal[False] = False) -> tuple[list[str], list[list[str]]] extract_categories_and_subcategories(categories: list[dict[str, list[str]] | str], flattened: Literal[True]) -> list[str] Extracts categories and subcategories from the `newsletter categories` dictionary in `newsletter settings`. Example for categories dict: [ {'main_category_1'}, {'main_category_2': ['sub_category_21', 'sub_category_22']} ] returning a tuple of lists: ['main_category_1', 'main_category_2'], [[], ['sub_category_21', 'sub_category_22']] if `flattened` is True, it returns a flat list of the above tuple: ['main_category_1', 'main_category_2', 'sub_category_21', 'sub_category_22'] .. py:function:: format_phone_number(phone_number: str) -> str Returns phone number in the international format +41 79 123 45 67