import numpy as np, pandas as pd
from typing import Dict, Tuple
from math import sqrt

rng = np.random.default_rng(42)
C, H, W = 3, 12, 12
n = C * H * W
L_z = 3.0
trials = 1000
power_iters = 80
tol = 1e-07
SIGMA_INCEPTION = np.array([0.5, 0.5, 0.5], dtype=np.float64)
SIGMA_CLIP = np.array([0.26862954, 0.26130258, 0.27577711], dtype=np.float64)


def make_D_inv_sigma(std_vec: np.ndarray) -> np.ndarray:
    inv_sigma_per_channel = 1.0 / std_vec.astype(np.float64)
    inv_sigmas = np.concatenate(
        [np.full(H * W, inv_sigma_per_channel[c]) for c in range(C)]
    )
    return inv_sigmas


def power_iteration_linop(
    matvec, rmatvec, dim: int, iters: int = 100, tol: float = 1e-07
) -> float:
    v = rng.standard_normal(dim)
    v = v / np.linalg.norm(v)
    sv_old = 0.0
    for _ in range(iters):
        w = matvec(v)
        if np.allclose(w, 0):
            return 0.0
        sv = np.linalg.norm(w)
        u = w / sv
        z = rmatvec(u)
        if np.allclose(z, 0):
            return 0.0
        v = z / np.linalg.norm(z)
        if abs(sv - sv_old) < tol * max(1.0, sv_old):
            break
        sv_old = sv
    return sv


def build_A_identity(L: float):
    def matvec(v):
        return L * v

    def rmatvec(v):
        return L * v

    return matvec, rmatvec


def build_A_random(L: float, dim: int):
    A = rng.standard_normal((dim, dim)).astype(np.float64) / sqrt(dim)

    def m(v):
        return A @ v

    def rt(v):
        return A.T @ v

    s = power_iteration_linop(m, rt, dim, iters=120, tol=1e-08)
    A *= L / s

    def matvec(v):
        return A @ v

    def rmatvec(v):
        return A.T @ v

    return matvec, rmatvec


def compose_with_D(matvec_A, rmatvec_A, inv_sigmas: np.ndarray):
    def matvec(x):
        return matvec_A(inv_sigmas * x)

    def rmatvec(y):
        return inv_sigmas * rmatvec_A(y)

    return matvec, rmatvec


ETAS = rng.standard_normal((trials, n))


def monte_carlo_ratio(matvec, dim: int):
    norms = []
    for i in range(trials):
        eta = ETAS[i]
        y = matvec(eta)
        norms.append(np.linalg.norm(y) / np.linalg.norm(eta))
    norms = np.array(norms)
    return float(norms.mean()), float(norms.max())


def run_with_ops(
    A_label: str, matvec_A, rmatvec_A, std_name: str, std_vec: np.ndarray
) -> Dict:
    inv_sigmas = make_D_inv_sigma(std_vec)
    sigma_min = float(std_vec.min())
    bound = L_z / sigma_min
    matvec_AD, rmatvec_AD = compose_with_D(matvec_A, rmatvec_A, inv_sigmas)
    s_emp = power_iteration_linop(matvec_AD, rmatvec_AD, n, iters=power_iters, tol=tol)
    mc_mean, mc_max = monte_carlo_ratio(matvec_AD, n)
    return {
        "A": A_label,
        "Norm": std_name,
        "sigma_min": sigma_min,
        "Theoretical bound Lz/sigma_min": bound,
        "Measured ||A D||_2 (power-iter)": s_emp,
        "MC mean ratio": mc_mean,
        "MC max ratio": mc_max,
        "Bound / Measured": bound / s_emp if s_emp > 0 else np.inf,
        "Bound OK (<=1+1e-5)": s_emp <= bound * (1 + 1e-05),
    }


def main():
    rows = []
    A_I = build_A_identity(L_z)
    rows.append(run_with_ops("Identity(L*I)", *A_I, "INCEPTION", SIGMA_INCEPTION))
    rows.append(run_with_ops("Identity(L*I)", *A_I, "CLIP", SIGMA_CLIP))
    A_R = build_A_random(L_z, n)
    rows.append(run_with_ops("Random A (||A||=Lz)", *A_R, "INCEPTION", SIGMA_INCEPTION))
    rows.append(run_with_ops("Random A (||A||=Lz)", *A_R, "CLIP", SIGMA_CLIP))
    df = pd.DataFrame(rows)

    def add_ratio_rows(df: pd.DataFrame) -> pd.DataFrame:
        out = df.copy()
        for Aname in out["A"].unique():
            subset = out[out["A"] == Aname].set_index("Norm")
            if {"INCEPTION", "CLIP"}.issubset(subset.index):
                ratio_pred = (
                    subset.loc["CLIP", "Theoretical bound Lz/sigma_min"]
                    / subset.loc["INCEPTION", "Theoretical bound Lz/sigma_min"]
                )
                ratio_meas = (
                    subset.loc["CLIP", "Measured ||A D||_2 (power-iter)"]
                    / subset.loc["INCEPTION", "Measured ||A D||_2 (power-iter)"]
                )
                out.loc[
                    (out["A"] == Aname) & (out["Norm"] == "CLIP"),
                    "CLIP/INCEPTION predicted ratio",
                ] = ratio_pred
                out.loc[
                    (out["A"] == Aname) & (out["Norm"] == "CLIP"),
                    "CLIP/INCEPTION measured ratio",
                ] = ratio_meas
        return out

    df = add_ratio_rows(df)
    with pd.option_context("display.precision", 6):
        print(
            df[
                [
                    "A",
                    "Norm",
                    "sigma_min",
                    "Theoretical bound Lz/sigma_min",
                    "Measured ||A D||_2 (power-iter)",
                    "MC mean ratio",
                    "MC max ratio",
                    "Bound / Measured",
                    "Bound OK (<=1+1e-5)",
                    "CLIP/INCEPTION predicted ratio",
                    "CLIP/INCEPTION measured ratio",
                ]
            ]
        )


if __name__ == "__main__":
    main()
