import numpy as np
from scipy.stats import multivariate_normal, multivariate_t


def _jitter_spd(mat, eps=1e-8):
    """Add a tiny diagonal jitter to reduce SciPy PSD warnings."""
    mat = np.asarray(mat, dtype=float)
    mat = 0.5 * (mat + mat.T)
    return mat + eps * np.eye(mat.shape[0])


   
def dgpK5_unb(n, p, ratio, param, null_type):
    epsilon = 0.1
    X = multivariate_normal.rvs(mean=np.zeros(p), cov=np.eye(p), size=round(ratio * n))
    n2 = round((1 - ratio) * n)
    n_perturbed = max(1, round(epsilon * n2))
    
    if null_type == "mean":
        mean_vec = np.full(p, param)
        Y = multivariate_normal.rvs(mean=np.zeros(p), cov=np.eye(p), size=n2)
        perturbed_indices = np.random.choice(n2, size=n_perturbed, replace=False)
        Y[perturbed_indices, :] = multivariate_normal.rvs(
            mean=mean_vec, cov=np.eye(p), size=n_perturbed
        )
    
    elif null_type == "cov":
        cov_matrix = (1+param)*np.eye(p)
        cov_matrix = _jitter_spd(cov_matrix)
        Y = multivariate_normal.rvs(mean=np.zeros(p), cov=np.eye(p), size=n2)
        perturbed_indices = np.random.choice(n2, size=n_perturbed, replace=False)
        Y[perturbed_indices, :] = multivariate_normal.rvs(
            mean=np.zeros(p), cov=cov_matrix, size=n_perturbed
        )
    
    elif null_type == "loc":
        mean_vec = np.full(p, param/2)
        cov_matrix = (1+param)*np.eye(p)
        cov_matrix = _jitter_spd(cov_matrix)
        Y = multivariate_normal.rvs(mean=np.zeros(p), cov=np.eye(p), size=n2)
        perturbed_indices = np.random.choice(n2, size=n_perturbed, replace=False)
        Y[perturbed_indices, :] = multivariate_normal.rvs(
            mean=mean_vec, cov=cov_matrix, size=n_perturbed
        )
    
    
    return X, Y


def dgpK6_unb(n, p, ratio, param, null_type):
    Sigma = np.zeros((p, p))
    for i in range(p):
        Sigma[i, :] = 0.4 ** np.abs(np.arange(1, p + 1) - (i + 1))
    Sigma = _jitter_spd(Sigma)
    
    X = multivariate_t.rvs(loc=np.zeros(p), shape=Sigma, size=round(ratio*n), df=10)
    
    if null_type == "mean":
        mean_vec = np.zeros(p)
        mean_vec[:p//2] = param
        Y = multivariate_t.rvs(loc=mean_vec, shape=Sigma, size=round((1-ratio)*n), df=10)
    elif null_type == "cov":
        Y = multivariate_t.rvs(loc=np.zeros(p), shape=(1+param)*Sigma, size=round((1-ratio)*n), df=10)
    elif null_type == "loc":
        mean_vec = np.zeros(p)
        mean_vec[:p//2] = param/2
        Y = multivariate_t.rvs(loc=mean_vec, shape=(1+param)*Sigma, size=round((1-ratio)*n), df=10)
    elif null_type == "dstr":
        Y = multivariate_t.rvs(loc=np.zeros(p), shape=Sigma, size=round((1-ratio)*n), df=param)
    return X, Y

