Source code for core.converters

""" Contains custom converters. """
from __future__ import annotations

import isodate
import morepath

from datetime import date, datetime
from onegov.core.framework import Framework
from onegov.core.orm.abstract import MoveDirection
from onegov.core.utils import is_uuid
from onegov.core.custom import custom_json as json
from time import mktime, strptime
from uuid import UUID


from typing import get_args, get_origin, overload, Any, Literal, TYPE_CHECKING
if TYPE_CHECKING:
    from collections.abc import Mapping
    from typing import LiteralString


@overload
[docs] def extended_date_decode(s: Literal['']) -> None: ... # type:ignore
@overload def extended_date_decode(s: str) -> date: ... def extended_date_decode(s: str) -> date | None: """ Decodes a date string HTML5 (RFC3339) compliant.""" if not s: return None try: return date.fromtimestamp(mktime(strptime(s, '%Y-%m-%d'))) except OverflowError as exception: raise ValueError() from exception
[docs] def extended_date_encode(d: date | None) -> str: """ Encodes a date HTML5 (RFC3339) compliant. """ if not d: return '' return d.strftime('%Y-%m-%d')
[docs] extended_date_converter = morepath.Converter( decode=extended_date_decode, encode=extended_date_encode )
@overload # type:ignore[overload-overlap]
[docs] def json_decode(s: Literal['']) -> None: ...
@overload def json_decode(s: str) -> dict[str, Any]: ... # NOTE: Technically this is incorrect, but we assume, we only ever # decode a JSON object, and not JSON in general def json_decode(s: str) -> dict[str, Any] | None: """ Decodes a json string to a dict. """ if not s: return None return json.loads(s)
[docs] def json_encode(d: Mapping[str, Any] | None) -> str: """ Encodes a dictionary to json. """ if not d: return '{}' return json.dumps(d)
[docs] json_converter = morepath.Converter( decode=json_decode, encode=json_encode )
[docs] def uuid_decode(s: str) -> UUID | None: """ Turns a uuid string into a UUID instance. """ return is_uuid(s) and UUID(s) or None
[docs] def uuid_encode(uuid: UUID | str | None) -> str: """ Turns a UUID instance into a uuid string. """ if not uuid: return '' if isinstance(uuid, str): return uuid return uuid.hex
[docs] uuid_converter = morepath.Converter( decode=uuid_decode, encode=uuid_encode )
@Framework.converter(type=UUID)
[docs] def get_default_uuid_converter() -> morepath.Converter[UUID]: return uuid_converter
@overload
[docs] def bool_decode(s: Literal['0', '']) -> Literal[False]: ...
@overload def bool_decode(s: Literal['1']) -> Literal[True]: ... @overload def bool_decode(s: str) -> bool: ... def bool_decode(s: str) -> bool: """ Decodes a boolean. """ return not (s == '0' or s == '') @overload
[docs] def bool_encode(d: Literal[False] | None) -> Literal['0']: ...
@overload def bool_encode(d: Literal[True]) -> Literal['1']: ... @overload def bool_encode(d: bool | None) -> Literal['0', '1']: ... def bool_encode(d: bool | None) -> Literal['0', '1']: """ Encodes a boolean. """ return d and '1' or '0'
[docs] bool_converter: morepath.Converter[bool] = morepath.Converter( decode=bool_decode, encode=bool_encode )
@Framework.converter(type=bool)
[docs] def get_default_bool_converter() -> morepath.Converter[bool]: return bool_converter
@overload
[docs] def datetime_decode(s: Literal['']) -> None: ... # type:ignore
@overload def datetime_decode(s: str) -> datetime: ... def datetime_decode(s: str) -> datetime | None: """ Decodes a datetime. """ return None if not s else isodate.parse_datetime(s)
[docs] def datetime_encode(d: datetime | None) -> str: """ Encodes a datetime. """ return isodate.datetime_isoformat(d) if d else ''
[docs] datetime_converter = morepath.Converter( decode=datetime_decode, encode=datetime_encode )
[docs] def datetime_year_decode(s: str) -> int: """ Decodes a year limited to the range datetime provides. """ year = int(s) if datetime.min.year <= year <= datetime.max.year: return year raise ValueError('year outside valid range')
[docs] def datetime_year_encode(y: int | None) -> str: """ Encodes a year. """ return str(y) if y is not None else ''
[docs] datetime_year_converter = morepath.Converter( decode=datetime_year_decode, encode=datetime_year_encode )
@Framework.converter(type=datetime)
[docs] def get_default_datetime_converter() -> morepath.Converter[datetime]: return datetime_converter
[docs] def integer_range_decode(s: str) -> tuple[int, int] | None: if not s: return None s, _, e = s.partition('-') return int(s), int(e)
[docs] def integer_range_encode(t: tuple[int, int] | None) -> str: return t and f'{t[0]}-{t[1]}' or ''
[docs] integer_range_converter = morepath.Converter( decode=integer_range_decode, encode=integer_range_encode )
[docs] def move_direction_decode(s: str) -> MoveDirection | None: try: return MoveDirection[s] except KeyError: return None
# we are slightly more lax and allow arbitrary str values when encoding # so we can provide e.g. a template string that gets replaced later on
[docs] def move_direction_encode(d: str | MoveDirection | None) -> str: if d is None: return '' elif isinstance(d, str): return d return d.name
[docs] move_direction_converter = morepath.Converter( decode=move_direction_decode, encode=move_direction_encode )
@Framework.converter(type=MoveDirection)
[docs] def get_default_move_direction_converter( ) -> morepath.Converter[MoveDirection]: return move_direction_converter
if TYPE_CHECKING:
[docs] LiteralConverterBase = morepath.Converter[LiteralString]
else: LiteralConverterBase = morepath.Converter
[docs] class LiteralConverter(LiteralConverterBase): """ This is a ``Converter`` counter-part to ``typing.Literal``. """ # TODO: This should use TypeForm eventually @overload def __init__(self, literal_type: Any, /) -> None: ... @overload def __init__(self, *literals: LiteralString) -> None: ... def __init__(self, *literals: Any) -> None: if len(literals) == 1 and get_origin(literals[0]) is Literal: literals = get_args(literals[0]) if not all(isinstance(v, str) for v in literals): # TODO: Consider supporting float/int literals via their # respective converters in the future raise ValueError('We only support string literals for simplicity')
[docs] self.allowed_values: set[str] = set(literals)
[docs] def single_decode(self, string: str) -> str | None: return string if string in self.allowed_values else None
[docs] def single_encode(self, value: str | None) -> str: # NOTE: If we ever support non-string literals we may need to actually # encode them here. return str(value) if value is not None else ''