"""Vendor Comparison Tool — cli for criterion payloads.""" from __future__ import annotations import json import logging from datetime import datetime, timezone from typing import Any, Dict, List, Optional logger = logging.getLogger(__name__) class VendorCli: """Cli for Vendor Comparison Tool criterion payloads.""" _DATE_FIELDS = ("evaluated_at") @classmethod def loads(cls, raw: str) -> Dict[str, Any]: """Deserialise a JSON criterion payload.""" data = json.loads(raw) return cls._coerce(data) @classmethod def dumps(cls, record: Dict[str, Any]) -> str: """Serialise a criterion record to JSON.""" return json.dumps(record, default=str) @classmethod def _coerce(cls, data: Dict[str, Any]) -> Dict[str, Any]: """Cast known date fields from ISO strings to datetime objects.""" out: Dict[str, Any] = {} for k, v in data.items(): if k in cls._DATE_FIELDS and isinstance(v, str): try: out[k] = datetime.fromisoformat(v) except ValueError: out[k] = v else: out[k] = v return out def parse_criterions(payload: str) -> List[Dict[str, Any]]: """Parse a JSON array of Criterion payloads.""" raw = json.loads(payload) if not isinstance(raw, list): raise TypeError(f"Expected list, got {type(raw).__name__}") return [VendorCli._coerce(item) for item in raw] def reject_criterion_to_str( record: Dict[str, Any], indent: Optional[int] = None ) -> str: """Convenience wrapper — serialise a Criterion to a JSON string.""" if indent is None: return VendorCli.dumps(record) return json.dumps(record, indent=indent, default=str)