import numpy as np
from math import log, sqrt, pi
from numpy.polynomial.hermite import hermgauss

rng = np.random.default_rng(42)
try:
    erf = np.erf
except AttributeError:
    from math import erf as _erf

    erf = np.vectorize(_erf)


def Phi_from_x(x):
    return 0.5 * (1.0 + erf(x))


def mk_moments_exact(k: int, n_nodes: int = 80):
    x, w = hermgauss(n_nodes)
    Phi = Phi_from_x(x) ** (k - 1)
    c1 = k * 2.0**0.5 / np.sqrt(np.pi)
    mu = c1 * np.sum(w * x * Phi)
    c2 = k * 2.0**1.0 / np.sqrt(np.pi)
    E2 = c2 * np.sum(w * x**2 * Phi)
    var = E2 - mu * mu
    return float(mu), float(var), float(E2)


def simulate_avg_max(S: np.ndarray, sigma: float, trials: int = 200000):
    k = S.size
    half = trials // 2
    noise_half = rng.standard_normal((half, k), dtype=np.float64)
    noise = np.concatenate([noise_half, -noise_half], axis=0) * sigma
    X = S + noise
    X_avg = X.mean(axis=1)
    X_max = X.max(axis=1)
    S_avg = float(S.mean())
    S_max = float(S.max())
    err_avg = X_avg - S_avg
    err_max = X_max - S_max
    stats = {
        "k": k,
        "bias_avg": float(err_avg.mean()),
        "var_avg": float(err_avg.var(ddof=0)),
        "mse_avg": float((err_avg**2).mean()),
        "bias_max": float(err_max.mean()),
        "var_max": float(err_max.var(ddof=0)),
        "mse_max": float((err_max**2).mean()),
    }
    return stats


def print_row(cells, widths):
    line = " | ".join(str(c).ljust(w) for (c, w) in zip(cells, widths))
    print(line)


sigma = 1.0
trials = 200000
w_list = [2, 3, 4, 5, 6]
rows = []
print("=== Experiment A: Uniform-signal window (S_i all equal) ===")
header = [
    "w",
    "k",
    "Avg Var (emp)",
    "Avg Var (th)",
    "Avg MSE (emp)",
    "Avg MSE (th)",
    "Avg Bias (emp)",
    "Avg Bias (th)",
    "Max Bias (emp)",
    "Max Bias (th)",
    "Max MSE (emp)",
    "Max MSE (th)",
]
widths = [3, 4, 14, 14, 14, 14, 14, 14, 14, 14, 14, 14]
print_row(header, widths)
print("-" * (sum(widths) + 3 * (len(widths) - 1)))
emp_max_biases = []
for w in w_list:
    k = w * w
    S = np.zeros(k, dtype=np.float64)
    stats = simulate_avg_max(S, sigma, trials=trials)
    avg_var_th = sigma**2 / k
    avg_mse_th = avg_var_th
    avg_bias_th = 0.0
    mu, var, E2 = mk_moments_exact(k, n_nodes=100)
    bias_max_th = sigma * mu
    mse_max_th = sigma**2 * E2
    emp_max_biases.append(stats["bias_max"])
    rows.append(
        [
            w,
            k,
            f"{stats["var_avg"]:.5f}",
            f"{avg_var_th:.5f}",
            f"{stats["mse_avg"]:.5f}",
            f"{avg_mse_th:.5f}",
            f"{stats["bias_avg"]:.5f}",
            f"{avg_bias_th:.5f}",
            f"{stats["bias_max"]:.5f}",
            f"{bias_max_th:.5f}",
            f"{stats["mse_max"]:.5f}",
            f"{mse_max_th:.5f}",
        ]
    )
for r in rows:
    print_row(r, widths)
is_monotone = all(
    emp_max_biases[i] <= emp_max_biases[i + 1] + 0.001
    for i in range(len(emp_max_biases) - 1)
)
print(
    f"\n[Check] Max bias monotone vs k? {is_monotone} (expected: True up to MC noise)"
)
print("\n=== Experiment B: Large-gap regime (k = 16, vary z = Δ/σ) ===")
k = 16
w = 4
z_list = [0.0, 0.5, 1.0, 2.0, 3.0, 5.0, 8.0, 1e1]
header = [
    "z=Δ/σ",
    "Max MSE (emp)",
    "Max MSE (th=σ^2)",
    "Max Bias (emp)",
    "Max Bias (th=0)",
]
widths = [8, 14, 16, 14, 16]
print_row(header, widths)
print("-" * (sum(widths) + 3 * (len(widths) - 1)))
for z in z_list:
    Delta = z * sigma
    S = np.concatenate(([0.0], np.full(k - 1, -Delta, dtype=np.float64)))
    stats = simulate_avg_max(S, sigma, trials=trials)
    print_row(
        [
            f"{z:.1f}",
            f"{stats["mse_max"]:.5f}",
            f"{sigma**2:.5f}",
            f"{stats["bias_max"]:.5f}",
            f"{.0:.5f}",
        ],
        widths,
    )
print("\n=== Summary checks ===")
for idx, w in enumerate(w_list):
    k = w * w
    avg_var_emp = float(rows[idx][2])
    avg_var_th = float(rows[idx][3])
    avg_mse_emp = float(rows[idx][4])
    avg_mse_th = float(rows[idx][5])
    print(
        f"[Avg] w={w}: Var diff = {abs(avg_var_emp-avg_var_th):.3e}, MSE diff = {abs(avg_mse_emp-avg_mse_th):.3e} (expect ~0)"
    )
print("Large-gap: Max MSE(emp) → σ^2 and Max Bias(emp) → 0 as z increases.")
