import numpy as np

def robust_mu_sigma_from_tails(X, L=50, z=3.0):
    """
    Robustly estimate global equilibrium value mu* from tail means.
    Filters out clear non-converged tails using MAD.
    """
    tail = X[:, -L:]
    tail_means = tail.mean(axis=1)

    mu_med = np.median(tail_means)
    mad = np.median(np.abs(tail_means - mu_med))
    sigma_robust = 1.4826 * mad  # robust std estimate

    # keep only tails close to the median (likely converged)
    keep = np.abs(tail_means - mu_med) <= z * max(sigma_robust, 1e-12)
    mu_star = np.median(tail_means[keep]) if np.any(keep) else mu_med

    return mu_star, sigma_robust, keep, tail_means

def teq_per_traj_to_global_mu(X, mu_star, tol, W=10):
    """
    Earliest t where the next W points stay within tol of mu_star.
    Returns t_eq (=-1 if never).
    """
    n, T = X.shape
    t_eq = np.full(n, T, dtype=int)

    for i in range(n):
        d = np.abs(X[i] - mu_star)
        for t in range(0, T - W + 1):
            if d[t:t+W].max() <= tol:
                t_eq[i] = t
                break
    return t_eq



def cmpt_tq(X):
    """
    Compute t_eq per trajectory using robust mu* and sigma_robust.
    """
    mu_star, sigma_robust, keep, tail_means = robust_mu_sigma_from_tails(X, L=50, z=3.0)
    tol = max(0.05, 5.0 * sigma_robust)   # tune 2–4x typically
    T_eq = teq_per_traj_to_global_mu(X, mu_star, tol)
    return T_eq

