Source code for imednet.utils.validators

from __future__ import annotations

from datetime import datetime
from typing import Any, Callable, Dict, List, TypeVar

from imednet.utils.dates import parse_iso_datetime  # Centralized date parsing

T = TypeVar("T")

# Pre-computed sets for boolean parsing optimization
_TRUE_LOWER = {"true", "1", "yes", "y", "t"}
_FALSE_LOWER = {"false", "0", "no", "n", "f"}
# Include common casing variants to avoid .strip().lower() allocation in hot paths
_TRUE_VARIANTS = _TRUE_LOWER | {"True", "TRUE"}
_FALSE_VARIANTS = _FALSE_LOWER | {"False", "FALSE"}


# Sentinel value for empty timestamps to avoid allocation overhead
_SENTINEL_DATETIME = datetime(1969, 4, 20, 16, 20)


[docs]def parse_datetime(v: str | datetime) -> datetime: """Parse an ISO datetime string or return a sentinel value. The SDK historically returns ``datetime(1969, 4, 20, 16, 20)`` when a timestamp field is empty. This helper mirrors that behaviour for backward compatibility. """ if not v: return _SENTINEL_DATETIME if isinstance(v, str): return parse_iso_datetime(v) return v
[docs]def parse_bool(v: Any) -> bool: """ Normalize boolean values from various representations. Accepts bool, str, int, float and returns a bool. """ if isinstance(v, bool): return v if isinstance(v, str): # Optimized path for common API responses to avoid string manipulation if v in _TRUE_VARIANTS: return True if v in _FALSE_VARIANTS: return False # Fallback for irregular casing or whitespace val = v.strip().lower() if val in _TRUE_LOWER: return True if val in _FALSE_LOWER: return False # Fallback: Try float conversion (for "1.0", "inf", "nan") # Optimization: Only attempt if it looks numeric or special float # to avoid try/except overhead on common strings like "apple" if val and (val[0].isdigit() or val[0] in ("-", "+", ".") or val[0] in ("n", "i")): try: return bool(float(val)) except (ValueError, TypeError): pass if isinstance(v, (int, float)): return bool(v) return False
[docs]def parse_int_or_default(v: Any, default: int = 0, strict: bool = False) -> int: """ Normalize integer values, defaulting if None or empty string. If strict=True, raise ValueError on parse failure. """ if v is None or v == "": return default try: return int(v) except (ValueError, TypeError): # Fallback: Try float conversion (for "123.0", 5.0) try: return int(float(v)) except (ValueError, TypeError, OverflowError): if strict: raise return default
[docs]def parse_str_or_default(v: Any, default: str = "") -> str: """ Normalize string values, defaulting if None. """ return default if v is None else str(v)
[docs]def parse_list_or_default(v: Any, default_factory: Callable[[], List[T]] = list) -> List[T]: """ Normalize list values, defaulting if None. Ensures result is a list. """ if v is None: return default_factory() if isinstance(v, list): return v return [v]
[docs]def parse_dict_or_default( v: Any, default_factory: Callable[[], Dict[str, Any]] = dict ) -> Dict[str, Any]: """ Normalize dictionary values, defaulting if None. Ensures result is a dict. """ if v is None: return default_factory() if isinstance(v, dict): return v return default_factory() # fallback if not a dict