import os
import re
import json
import glob
from typing import Dict, Any, List

def read_json(p: str) -> dict:
    with open(p, "r", encoding="utf-8") as f:
        return json.load(f)

def write_json(p: str, obj: dict):
    os.makedirs(os.path.dirname(p), exist_ok=True)
    with open(p, "w", encoding="utf-8") as f:
        json.dump(obj, f, ensure_ascii=False, indent=2)

def pred_path_for_features(features_path: str) -> str:
    base = os.path.basename(features_path)
    return os.path.join(os.path.dirname(features_path), base.replace("features_", "pred_"))

def label_path_for_features(features_path: str) -> str:
    base = os.path.basename(features_path)
    return os.path.join(os.path.dirname(features_path), base.replace("features_", "label_"))

def extract_json_block(text: str) -> dict | None:
    try:
        return json.loads(text)
    except Exception:
        pass
    m = re.search(r"\{[\s\S]*\}", text)
    if m:
        try:
            return json.loads(m.group(0))
        except Exception:
            return None
    return None

def filter_pred_names_to_step(features: dict, op_names: List[str]) -> List[str]:
    valid = {op["name"] for op in (features.get("ops_prob") or []) if "name" in op}
    out, seen = [], set()
    for n in (op_names or []):
        if n in valid and n not in seen:
            out.append(n)
            seen.add(n)
    return out

def augment_with_parents(features: dict, names: List[str]) -> List[str]:
    parent_map = {}
    valid = []
    for op in features.get("ops_prob", []):
        nm = op.get("name")
        pr = op.get("parent")
        if nm is not None:
            parent_map[nm] = pr
            valid.append(nm)
    valid_set = set(valid)
    s = set(names or [])
    for n in list(s):
        p = parent_map.get(n)
        if p and p in valid_set:
            s.add(p)
    ordered = []
    for n in (names + valid):
        if n in s and n not in ordered:
            ordered.append(n)
    return ordered

def iter_feature_files(root_dir: str) -> List[str]:
    pattern = os.path.join(root_dir, "**", "features_[hv]_step_*.json")
    return sorted(glob.glob(pattern, recursive=True))
