import numpy as np
from numpy.fft import fft2, ifft2


def compute_gamma(k, sigma=0.1, size=None, trials=100, seed=0, center_kernel=True):
    if size is None:
        size = max(256, 8 * k)
    rng = np.random.default_rng(seed)
    kernel = np.ones((k, k), dtype=np.float64) / k**2
    ker_full = np.zeros((size, size), dtype=np.float64)
    i0 = (size - k) // 2
    j0 = (size - k) // 2
    ker_full[i0 : i0 + k, j0 : j0 + k] = kernel
    if center_kernel:
        ker_full = np.fft.ifftshift(ker_full)
    kernel_fft = fft2(ker_full)
    gammas = []
    for _ in range(trials):
        eta = rng.normal(0.0, sigma, size=(size, size))
        y_c = ifft2(fft2(eta) * kernel_fft)
        assert (
            np.max(np.abs(np.imag(y_c))) < 1e-10
        ), "Significant imaginary part detected"
        y = y_c.real
        norm_sq = np.sum(y**2, dtype=np.float64)
        gamma = norm_sq / (sigma**2 * size**2)
        gammas.append(gamma)
    emp_gamma = float(np.mean(gammas))
    std = float(np.std(gammas, ddof=1))
    theo_gamma = float(np.sum(kernel**2))
    return emp_gamma, std, theo_gamma


ks = [4, 8, 12, 16, 20, 24, 28, 32]
N = 100
for k in ks:
    emp, std, theo = compute_gamma(k, sigma=0.1, trials=N, seed=42)
    ratio = emp / theo if theo > 0 else np.nan
    rel_err = abs(emp - theo) / theo
    se = std / np.sqrt(N)
    within_se = abs(emp - theo) <= 3 * se
    tol = 0.03
    print(f"For k={k}:")
    print(f" Empirical γ = {emp:.6f} ± {std:.6f} (SE {se:.6f})")
    print(f" Theoretical γ = {theo:.6f} (box filter ⇒ 1/k^2)")
    print(f" Ratio (empirical / theoretical) = {ratio:.6f}")
    print(f" Matches within 3%? {"Yes"if rel_err<=tol else"No"}")
    print(f" Matches within 3 SE? {"Yes"if within_se else"No"}\n")
