"""Power-law fitting for V_opt(c) ~ K * c^{-alpha}."""
from __future__ import annotations
import numpy as np
import pandas as pd
from dataclasses import dataclass
from scipy import stats

@dataclass
class PowerLawFit:
    alpha: float
    K: float
    r2: float
    slope: float
    intercept: float

def fit_power_law(c: np.ndarray, vopt: np.ndarray, eps: float=1e-12) -> PowerLawFit:
    # Fit log(vopt) = logK - alpha*logc
    mask = (c > 0) & (vopt > eps) & np.isfinite(c) & np.isfinite(vopt)
    c = c[mask]; vopt = vopt[mask]
    if len(c) < 2:
        return PowerLawFit(alpha=float("nan"), K=float("nan"), r2=float("nan"), slope=float("nan"), intercept=float("nan"))
    x = np.log(c)
    y = np.log(vopt)
    slope, intercept, r, p, se = stats.linregress(x, y)
    alpha = -slope
    K = float(np.exp(intercept))
    r2 = float(r**2)
    return PowerLawFit(alpha=float(alpha), K=float(K), r2=r2, slope=float(slope), intercept=float(intercept))

def fit_power_law_by_group(curve: pd.DataFrame, group_cols: list[str], c_col: str="probe_c", v_col: str="V_opt") -> pd.DataFrame:
    rows = []
    for key, g in curve.groupby(group_cols, dropna=False):
        c = g[c_col].to_numpy()
        v = g[v_col].to_numpy()
        fit = fit_power_law(c, v)
        if not isinstance(key, tuple):
            key = (key,)
        row = {col: val for col, val in zip(group_cols, key)}
        row.update({"alpha_hat": fit.alpha, "K_hat": fit.K, "r2": fit.r2})
        rows.append(row)
    return pd.DataFrame(rows)
