import os, csv
from typing import List, Tuple, Dict, Sequence

MSPair = Tuple[Tuple[int, ...], float]
VecPair = Tuple[Tuple[int, ...], float]
def append_curve_ms_csv(path: str, q_vec: Sequence[int], v: float):
    os.makedirs(os.path.dirname(path) or ".", exist_ok=True)
    exist = os.path.exists(path)
    with open(path, "a", newline="") as f:
        w = csv.writer(f)
        if not exist:
            S = len(q_vec)
            w.writerow([*(f"q{i+1}" for i in range(S)), "V"])
        w.writerow([*map(int, q_vec), float(v)])

def load_curve_ms_csv(path: str) -> List[MSPair]:
    if not os.path.exists(path):
        raise FileNotFoundError(f"curve_csv not found: {path}")
    pts: List[MSPair] = []
    with open(path, "r", newline="") as f:
        reader = csv.DictReader((row for row in f if not row.lstrip().startswith("#")))
        if "V" not in reader.fieldnames:
            raise ValueError("curve_csv must include column 'V' and one or more 'qk' columns")
        qcols = [c for c in reader.fieldnames if c.lower().startswith("q")]
        if not qcols:
            raise ValueError("curve_csv must include at least one q-column: q1, [q2,...]")
        for row in reader:
            q = tuple(int(row[c]) for c in qcols)
            v = float(row["V"])
            if all(qq > 0 for qq in q):
                pts.append((q, v))
    seen: Dict[Tuple[int, ...], float] = {}
    for q, v in pts:
        seen[q] = v
    return [(q, seen[q]) for q in sorted(seen.keys())]

def load_curve_ms_csv_or_die(path: str) -> List[MSPair]:
    pts = load_curve_ms_csv(path)
    if not pts:
        raise ValueError("curve_ms.csv has no valid rows.")
    S = len(pts[0][0])
    for q,_ in pts:
        if len(q) != S:
            raise ValueError("Inconsistent S in curve file.")
    return pts

def load_curve_csv_strict_multi(path: str, K: int)->List[VecPair]:
    if not os.path.exists(path):
        raise FileNotFoundError(f"curve_csv not found: {path}")
    pts = []
    with open(path, "r", newline="") as f:
        reader = csv.DictReader((row for row in f if not row.lstrip().startswith("#")))
        need = [f"q{i+1}" for i in range(K)]
        if any(col not in reader.fieldnames for col in need) or "V" not in reader.fieldnames:
            raise ValueError(f"curve_csv must have columns {need+['V']}. found={reader.fieldnames}")
        for row in reader:
            q_vec = tuple(int(row[f"q{i+1}"]) for i in range(K))
            V = float(row["V"])
            if all(q>0 for q in q_vec):
                pts.append((q_vec, V))
    pts.sort(key=lambda x: x[0])
    return pts