import os
import json
from typing import Dict, Any, List, Optional


def _load_basic_info(path: str) -> Optional[Dict[str, Any]]:
    try:
        with open(path, "r", encoding="utf-8") as f:
            return json.load(f)
    except Exception:
        return None


def summarize_candidates(workspace_path: str) -> Dict[str, Any]:
    """
    Summarize candidates under <workspace_path>/candidates.

    For each candidate, compute deltas vs parent and a success flag:
      success := (acc_child > acc_parent) or (acc_child == acc_parent and cost_child < cost_parent)

    Returns a dict and also writes:
      - <workspace_path>/summary.json
      - <workspace_path>/candidates/candidate_<n>/optimization_result.json
    """
    cdir = os.path.join(workspace_path, "candidates")
    result: Dict[str, Any] = {
        "workspace_path": workspace_path,
        "candidates": [],
        "edges": [],
        "best": None,
    }

    if not os.path.isdir(cdir):
        with open(os.path.join(workspace_path, "summary.json"), "w", encoding="utf-8") as f:
            json.dump(result, f, ensure_ascii=False, indent=2)
        return result

    # Load basics
    basics_by_round: Dict[int, Dict[str, Any]] = {}
    for name in sorted(os.listdir(cdir)):
        if not name.startswith("candidate_"):
            continue
        try:
            r = int(name.split("_")[-1])
        except Exception:
            continue
        info = _load_basic_info(os.path.join(cdir, name, "basic_info.json")) or {}
        info["folder_name"] = name
        basics_by_round[r] = info

    # Build summaries
    def _m(info: Dict[str, Any], key: str) -> Optional[float]:
        try:
            val = (info.get("metrics") or {}).get(key)
            return None if val is None else float(val)
        except Exception:
            return None

    best = {"round": None, "accuracy": -1.0, "cost": None}
    for r in sorted(basics_by_round.keys()):
        info = basics_by_round[r]
        parent = info.get("parent")
        acc = _m(info, "accuracy")
        cost = _m(info, "cost")
        parent_acc = None
        parent_cost = None
        acc_delta = None
        cost_delta = None
        success = None

        if parent is not None and parent in basics_by_round:
            pinfo = basics_by_round[parent]
            parent_acc = _m(pinfo, "accuracy")
            parent_cost = _m(pinfo, "cost")
            if parent_acc is not None and acc is not None:
                acc_delta = acc - parent_acc
            if parent_cost is not None and cost is not None:
                cost_delta = cost - parent_cost
            # Success rule
            if acc is not None and parent_acc is not None:
                if acc > parent_acc:
                    success = True
                elif acc == parent_acc and (cost is not None and parent_cost is not None) and cost < parent_cost:
                    success = True
                else:
                    success = False
            else:
                success = False

            result["edges"].append([parent, r])

        # Update best
        if acc is not None and acc > best["accuracy"]:
            best = {"round": r, "accuracy": acc, "cost": cost}

        item = {
            "round": r,
            "parent": parent,
            "accuracy": acc,
            "cost": cost,
            "parent_accuracy": parent_acc,
            "parent_cost": parent_cost,
            "acc_delta": acc_delta,
            "cost_delta": cost_delta,
            "success": success,
            "trajectory_path": info.get("trajectory_path"),
        }
        result["candidates"].append(item)

        # Write per-candidate summary
        try:
            with open(os.path.join(cdir, info.get("folder_name"), "optimization_result.json"), "w", encoding="utf-8") as f:
                json.dump(item, f, ensure_ascii=False, indent=2)
        except Exception:
            pass

    result["best"] = best if best["round"] is not None else None

    # Write root summary
    try:
        with open(os.path.join(workspace_path, "summary.json"), "w", encoding="utf-8") as f:
            json.dump(result, f, ensure_ascii=False, indent=2)
    except Exception:
        pass

    return result

