"""Regime assignment utilities.

Regimes are assigned *post hoc* based on estimated (L_int, alpha) and are used
for analysis/visualization only (never for tuning).
"""
from __future__ import annotations
from dataclasses import dataclass
from typing import Dict, Any
import numpy as np
import pandas as pd

@dataclass(frozen=True)
class RegimeNames:
    static_sufficient: str = "Static-Sufficient"
    dynamic_critical: str = "Dynamic-Critical"
    noise_dominant: str = "Noise-Dominant"

NAMES = RegimeNames()

def thresholds_from_config(points: pd.DataFrame, cfg: Dict[str, Any]) -> Dict[str, float]:
    """Compute thresholds from config; supports fixed or quantile rules."""
    alpha = points["alpha_hat"].to_numpy()
    lint = points["L_int"].to_numpy()
    out: Dict[str, float] = {}

    mode = cfg.get("mode", "quantile")
    if mode == "fixed":
        out["alpha_hi"] = float(cfg["alpha_hi"])
        out["alpha_lo"] = float(cfg["alpha_lo"])
        out["lint_hi"] = float(cfg["lint_hi"])
        return out

    # quantile mode (default): deterministic and dataset-independent in code
    aq_hi = float(cfg.get("alpha_q_hi", 0.66))
    aq_lo = float(cfg.get("alpha_q_lo", 0.33))
    lq_hi = float(cfg.get("lint_q_hi", 0.75))

    out["alpha_hi"] = float(np.nanquantile(alpha, aq_hi))
    out["alpha_lo"] = float(np.nanquantile(alpha, aq_lo))
    out["lint_hi"] = float(np.nanquantile(lint, lq_hi))
    return out

def assign_regime(points: pd.DataFrame, cfg: Dict[str, Any]) -> pd.Series:
    """Assign regimes using (alpha_hat, L_int)."""
    thr = thresholds_from_config(points, cfg)
    alpha = points["alpha_hat"]
    lint = points["L_int"]

    reg = np.full(len(points), NAMES.dynamic_critical, dtype=object)
    reg[lint >= thr["lint_hi"]] = NAMES.noise_dominant
    # only classify as static-sufficient if not noise-dominant
    mask = lint < thr["lint_hi"]
    reg[np.where(mask & (alpha >= thr["alpha_hi"]))] = NAMES.static_sufficient
    # remaining are dynamic-critical
    return pd.Series(reg, index=points.index), thr
