from __future__ import annotations

import json
import os
import pickle
import hashlib
from typing import Any, Dict, Tuple


def _cache_key(func_name: str, payload: Dict[str, Any]) -> Tuple[str, str]:
    """ cache key for the given inputs."""
    payload_bytes = json.dumps(payload, sort_keys=True, default=str).encode("utf-8")
    digest = hashlib.md5(payload_bytes, usedforsecurity=False).hexdigest()
    key_full = f"{func_name}:{digest}"
    return digest[:10], key_full


def _cache_paths(cache_dir: str, func_name: str, key_short: str) -> Tuple[str, str]:
    """ cache paths for the given inputs."""
    base = os.path.join(cache_dir, f"{func_name}_{key_short}")
    return f"{base}.pkl", f"{base}.meta.json"


def cache_load(cache_dir: str, func_name: str, payload: Dict[str, Any]) -> Any:
    """Load cached experiment results if available."""
    os.makedirs(cache_dir, exist_ok=True)
    key_short, _ = _cache_key(func_name, payload)
    pkl_path, _meta_path = _cache_paths(cache_dir, func_name, key_short)
    if not os.path.exists(pkl_path):
        return None
    try:
        with open(pkl_path, "rb") as f:
            return pickle.load(f)
    except Exception:
        return None


def cache_save(cache_dir: str, func_name: str, payload: Dict[str, Any], obj: Any) -> None:
    """Save experiment results to cache."""
    os.makedirs(cache_dir, exist_ok=True)
    key_short, key_full = _cache_key(func_name, payload)
    pkl_path, meta_path = _cache_paths(cache_dir, func_name, key_short)
    with open(pkl_path, "wb") as f:
        pickle.dump(obj, f, protocol=pickle.HIGHEST_PROTOCOL)
    with open(meta_path, "w", encoding="utf-8") as f:
        json.dump({"key": key_full, "payload": payload}, f, indent=2, sort_keys=True)


def cache_version() -> str:
    """Return the cache format version."""
    return "v1"


def estimators_signature(estimators: Dict[str, Any]) -> Dict[str, Any]:
    """Create a signature for estimator configurations."""
    sig = {}
    for k, est in estimators.items():
        sig[k] = str(getattr(est, "config", None) or getattr(est, "__dict__", None))
    return sig


def jsonable(obj: Any) -> Any:
    """Convert objects to JSON-serializable representations."""
    try:
        json.dumps(obj)
        return obj
    except Exception:
        return str(obj)

