import numpy as np
import matplotlib.pyplot as plt
import torch
import torch.nn as nn
import torch.optim as optim
import torch.distributions as dist
from torchvision import datasets, transforms
from torch.utils.data import TensorDataset, DataLoader, Subset
from tqdm import tqdm
import copy
import warnings
warnings.filterwarnings("ignore", category=FutureWarning)

custom_colors = ['darkorange', '#ba2406', '#5B478B', '#356792']
plt.rcParams.update({
    'font.family': 'serif',
    'text.usetex': True,
    'font.size': 1,
    'axes.titlesize': 21,
    'axes.labelsize': 21,
    'legend.fontsize': 16,
    'xtick.labelsize': 16,
    'ytick.labelsize': 16,
    'figure.dpi': 400,
    'figure.figsize': [6, 4],      
    'axes.linewidth': 0.8,
    'lines.linewidth': 2.0,             
    'lines.markersize': 5,            
    'axes.grid': True,
    'grid.linestyle': '--',
    'grid.linewidth': 0.15,
    'legend.frameon': True,
    'legend.framealpha': 0.75,
    'legend.loc': 'best',
    'savefig.bbox': 'tight',
    'savefig.pad_inches': 0.02,
    'axes.spines.top': True,
    'axes.spines.right': True,
    'axes.prop_cycle': plt.cycler('color', custom_colors)
})

sigmoid = nn.Sigmoid()
logsigmoid = nn.LogSigmoid()
num_points = 5000
d = 200

def projection(q_xi, q_Xi, U, D):
    q_cov = q_Xi - q_xi * q_xi
    q_xi_new = q_xi.clamp(min=-U, max=U)
    q_cov_new = q_cov.clamp(min=1/D, max=D)
    return (q_xi_new, q_cov_new + q_xi_new * q_xi_new)

def prepare_dataset(X, y, batch):
    dataset = TensorDataset(X, y)
    data_loader = DataLoader(dataset, batch_size=batch, shuffle=True)
    return data_loader

def log_prob_logistic(X, y, samples):
    w = samples
    logits = X @ w.t()
    logits = logits.t()
    return logsigmoid(y.unsqueeze(0) * logits).sum(dim=1)

def elbo_gd(X, y, q_mu, q_cov_sq, prior, m):
    q_cov = q_cov_sq * q_cov_sq
    q_dist = dist.MultivariateNormal(q_mu, torch.diag_embed(q_cov))
    kl = dist.kl_divergence(q_dist, prior)
        
    eps = torch.randn((d, m))
    samples = q_mu.unsqueeze(1) + torch.diag_embed(q_cov_sq) @ eps
    log_probs = log_prob_logistic(X, y, samples.t())
        
    log_likelihood = (log_probs.mean()) * num_points / y.shape[0]
    
    entropy = q_dist.entropy()
        
    elbo = log_likelihood - kl
    free_energy = -log_likelihood + kl + entropy
    return elbo, kl, log_likelihood, free_energy

def elbo_ngd(X, y, q_xi, q_Xi, prior, m):
    q_cov = q_Xi - q_xi * q_xi
    q_dist = dist.MultivariateNormal(q_xi, torch.diag_embed(q_cov))
    kl = dist.kl_divergence(q_dist, prior)

    q_cov_sq = torch.sqrt(q_cov)
        
    eps = torch.randn((d, m))
    samples = q_xi.unsqueeze(1) + torch.diag_embed(q_cov_sq) @ eps
    log_probs = log_prob_logistic(X, y, samples.t())
    log_likelihood = (log_probs.mean()) * num_points / y.shape[0]

    elbo = log_likelihood - kl
    return elbo, kl, log_likelihood, samples
        
def true_parameter_gd(data, prior, m, q_mu, q_cov_sq, num_iter, lr, method, id):
    lr0 = lr
    optimizer = optim.SGD([q_mu, q_cov_sq], lr=lr)
    elbo_history = []
    for iteration in tqdm(range(1, 1 + num_iter)):
        for X, y in data:
            curr_lr = lr0 / np.sqrt(iteration)
            for param_group in optimizer.param_groups:
                param_group['lr'] = curr_lr
            optimizer.zero_grad()
            elbo, kl, log_likelihood, free_energy = elbo_gd(X, y, q_mu, q_cov_sq, prior, m)
            elbo_history.append(elbo.item())
            if method is not None:
                if method[0] == 'proj':
                    loss = -elbo
                if method[0] == 'prox':
                    loss = free_energy
            loss.backward()
            optimizer.step()
            with torch.no_grad():
                if method is not None:
                    if method[0] == 'proj':
                        q_cov_sq.clamp_(min=1 / torch.sqrt(method[1]))
                    if method[0] == 'prox':
                        current_lr = optimizer.param_groups[0]['lr']
                        q_cov_sq += (torch.sqrt(q_cov_sq * q_cov_sq + 4 * current_lr) - q_cov_sq) / 2
        
        if iteration % 10 == 0:
            print(f"Iteration {iteration}: ELBO = {elbo.item():.4f}, KL = {kl.item():.4f}, Log-Likelihood = {log_likelihood.item():.4f}")
            
    torch.save({'mu': q_mu.detach(), 'cov': (q_cov_sq * q_cov_sq).detach(), 'elbo_history': elbo_history, 'elbo': {elbo.detach()}}, f'model/madelon_gd_d={d}_lr={lr0}_method={method[0]}_id={id}.pt')
    return elbo_history, q_mu, q_cov_sq * q_cov_sq
    
def grad_ngd_ad(log_likelihood, q_xi, q_Xi):
    return torch.autograd.grad(log_likelihood, [q_xi, q_Xi])

def grad_ngd_bonnet(X, y, samples, q_xi):
    logits = (X @ samples).t()
    sig = sigmoid(-y.unsqueeze(0) * logits)
    sig_mean = sig.mean(dim=0)
    temp = y * sig_mean
    gd1 = X.t() @ temp
    
    val_mean = (sig * (1 - sig)).mean(dim=0)
    gd2 = -(X * X).t() @ val_mean
    
    gd_xi = gd1 - gd2 * q_xi
    gd_Xi = gd2 / 2
    gd_xi *= num_points / y.shape[0]
    gd_Xi *= num_points / y.shape[0]
    return [gd_xi, gd_Xi]
    
def true_parameter_ngd(data, prior, m, q_mu, q_cov_sq, num_iter, lr, proj, id):
    lr0 = lr
    eta_p_Lambda = -torch.ones(d) / 2
    eta_Lambda = -1 / (q_cov_sq * q_cov_sq) / 2
    eta_lambda = -2 * eta_Lambda * q_mu
    elbo_history = []
    q_xi = q_mu
    q_Xi = q_cov_sq * q_cov_sq + q_mu * q_mu
    for iteration in tqdm(range(1, 1 + num_iter)):
        for X, y in data:
            lr = lr0 / np.sqrt(iteration)
            elbo, kl, log_likelihood, samples = elbo_ngd(X, y, q_xi, q_Xi, prior, m)
            elbo_history.append(elbo.item())
            grad_omega = grad_ngd_bonnet(X, y, samples, q_xi)
                
            with torch.no_grad():
                eta_lambda = (1 - lr) * eta_lambda + lr * grad_omega[0]
                eta_Lambda = (1 - lr) * eta_Lambda + lr * (grad_omega[1] + eta_p_Lambda)
            q_cov = -1 / eta_Lambda / 2
            q_xi = q_cov * eta_lambda
            q_Xi = q_cov + q_xi * q_xi
            if proj is not None:
                with torch.no_grad():
                    q_xi, q_Xi = projection(q_xi, q_Xi, proj[0], proj[1])
            q_xi = q_xi.detach().requires_grad_(True)
            q_Xi = q_Xi.detach().requires_grad_(True)
        
        if iteration % 10 == 0:
            print(f"Iteration {iteration}: ELBO = {elbo.item():.4f}, KL = {kl.item():.4f}, Log-Likelihood = {log_likelihood.item():.4f}")
    
    torch.save({'mu': q_xi.detach(), 'cov': (q_Xi - q_xi * q_xi).detach(), 'elbo_history': elbo_history, 'elbo': {elbo.detach()}}, f'model/madelon_ngd_proj={proj}_d={d}_lr={lr0}_id={id}.pt')
    return elbo_history, q_xi, q_Xi - q_xi * q_xi

def experiment_madelon():
    file_x0 = 'data/madelon_train.data'
    file_x1 = 'data/madelon_valid.data'
    file_y0 = 'data/madelon_train.labels'
    file_y1 = 'data/madelon_valid.labels'

    def load_file(filepath):
        with open(filepath, 'r') as f:
            lines = f.readlines()
        data = [list(map(float, line.strip().split())) for line in lines if line.strip()]
        return np.array(data)

    X0 = load_file(file_x0)
    X1 = load_file(file_x1)
    Xs = np.vstack((X0, X1))
    y0 = load_file(file_y0)
    y1 = load_file(file_y1)
    ys = np.concatenate((y0, y1))
    
    Xs_tensor = torch.tensor(Xs, dtype=torch.float32)
    ys_tensor = torch.tensor(ys, dtype=torch.float32).squeeze()

    # Optional: Normalize features (mean=0, std=1 per column)
    mean = Xs_tensor.mean(dim=0, keepdim=True)
    std = Xs_tensor.std(dim=0, keepdim=True)
    Xs_tensor = (Xs_tensor - mean) / std
    
    data_loader = DataLoader(TensorDataset(Xs_tensor, ys_tensor), batch_size=2000, shuffle=True)
    global num_points
    num_points = Xs.shape[0]
    global d
    d = Xs.shape[1]

    num_iter = 1000
    m = 2000
    prior = dist.MultivariateNormal(torch.zeros(d), torch.eye(d))
    for id in range(5):
        q_mu0 = torch.nn.Parameter(torch.randn(d))
        q_cov_sq0 = torch.nn.Parameter(torch.exp(torch.randn(d)))

        for lr in [5e-1, 2e-1, 1e-1, 5e-2, 2e-2, 1e-2, 5e-3, 2e-3, 1e-3, 5e-4]:
            elbo_history, q_mu, q_cov = true_parameter_gd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, ('proj', torch.tensor([400])), id)
            elbo_history, q_mu, q_cov = true_parameter_gd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, ('prox', ), id)
            elbo_history, q_mu, q_cov = true_parameter_ngd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, None, id)
            elbo_history, q_mu, q_cov = true_parameter_ngd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, (3, 400), id)
            elbo_history, q_mu, q_cov = true_parameter_ngd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, (2, 300), id)
            elbo_history, q_mu, q_cov = true_parameter_ngd(data_loader, prior, m, copy.deepcopy(q_mu0), copy.deepcopy(q_cov_sq0), num_iter, lr, (1.5, 200), id)

def plot_results_convergence(mode):
    d = 500
    num_show = 500
    if mode == 0:
        gd_models = ['prox', 'proj']
        ngd_models = ['None', (3, 400)]
    elif mode == 1:
        gd_models = []
        ngd_models = ['None', (3, 400), (2, 300), (1.5, 200)]
    gd_lrs = [1e-2]
    ngd_lrs = [5e-1]
    results = {}
    for gd_model in gd_models:
        results[gd_model] = {} 
        for lr in gd_lrs:
            results[gd_model][lr] = []
            for id in range(5):
                results[gd_model][lr].append([-i for i in torch.load(f'model/madelon_gd_d={d}_lr={lr}_method={gd_model}_id={id}.pt')['elbo_history']])
            results[gd_model][lr] = np.array(results[gd_model][lr])
            results[gd_model][lr] = results[gd_model][lr].reshape(5, 1000, 2)
            results[gd_model][lr] = (10 * results[gd_model][lr][:, :, 0] + 3 * results[gd_model][lr][:, :, 1]) / 13
    for ngd_model in ngd_models:
        results[ngd_model] = {}
        for lr in ngd_lrs:
            results[ngd_model][lr] = []
            for id in range(5):
                results[ngd_model][lr].append([-i for i in torch.load(f'model/madelon_ngd_proj={ngd_model}_d={d}_lr={lr}_id={id}.pt')['elbo_history']])
            results[ngd_model][lr] = np.array(results[ngd_model][lr])
            results[ngd_model][lr] = results[ngd_model][lr].reshape(5, 1000, 2)
            results[ngd_model][lr] = (10 * results[ngd_model][lr][:, :, 0] + 3 * results[ngd_model][lr][:, :, 1]) / 13
            
    rep_indices = {}
    for model in gd_models:
        rep_indices[model] = {}
        for lr in gd_lrs:
            elbos_trimmed = results[model][lr][:, :num_show]
            elbo_lower = np.percentile(elbos_trimmed, 25, axis=0)
            elbo_upper = np.percentile(elbos_trimmed, 75, axis=0)
            median_elbo = np.median(elbos_trimmed, axis=0)
            diff_sum = np.sum(np.abs(elbos_trimmed - median_elbo), axis=1)
            rep_index = np.argmin(diff_sum)
            rep_indices[model][lr] = rep_index
            
            plot_elbo = elbos_trimmed[rep_index]
            
            iterations = np.arange(1, 1 + num_show)
            
            if model == 'proj':
                plt.plot(iterations, plot_elbo, label=rf'Proj-SGD')
            elif model == 'prox':
                plt.plot(iterations, plot_elbo, label=rf'Prox-SGD')
            plt.fill_between(iterations, elbo_lower[:num_show], elbo_upper[:num_show], alpha=0.3)
            
    for model in ngd_models:
        rep_indices[model] = {}
        for lr in ngd_lrs:
            elbos_trimmed = results[model][lr][:, :num_show]
            elbo_lower = np.percentile(elbos_trimmed, 25, axis=0)
            elbo_upper = np.percentile(elbos_trimmed, 75, axis=0)
            median_elbo = np.median(elbos_trimmed, axis=0)
            diff_sum = np.sum(np.abs(elbos_trimmed - median_elbo), axis=1)
            rep_index = np.argmin(diff_sum)
            rep_indices[model][lr] = rep_index
            
            plot_elbo = elbos_trimmed[rep_index]
            
            iterations = np.arange(1, 1 + num_show)
            
            if model == 'None':
                if mode == 0:
                    plt.plot(iterations, plot_elbo, label=rf'SNGD')
                elif mode == 1:
                    plt.plot(iterations, plot_elbo, label=rf'U=$\infty$, D=$\infty$')
            else:
                if mode == 0:
                    plt.plot(iterations, plot_elbo, label=rf'Proj-SNGD')
                elif mode == 1:
                    plt.plot(iterations, plot_elbo, label=f'U={model[0]}, D={model[1]}')
            plt.fill_between(iterations, elbo_lower[:num_show], elbo_upper[:num_show], alpha=0.3)
            
    if mode == 0:
        plt.ylim(top=10000, bottom=2500)
    elif mode == 1:
        plt.ylim(top=10000, bottom=2500)
    plt.xlabel('Epoch')
    plt.ylabel('Negative ELBO')
    plt.legend()
    plt.grid(True)
    plt.tight_layout()
    plt.savefig(f'figure/madelon convergence {mode+1}.pdf')
    plt.clf()
    
def plot_results_robustness(mode):
    d = 500
    def find_index(elbo_history, threshold):
        for i in range(len(elbo_history) // 2):
            if 10 * elbo_history[2*i] / 13 + 3 * elbo_history[2*i+1] / 13 > threshold:
                return 2*i
        return len(elbo_history)
    threshold = -3000
    if mode == 0:
        gd_models = ['prox', 'proj']
        ngd_models = ['None', (3, 400)]
    elif mode == 1:
        gd_models = []
        ngd_models = ['None', (3, 400), (2, 300), (1.5, 200)]
    results = {}
    lrs = [5e-1, 2e-1, 1e-1, 5e-2, 2e-2, 1e-2, 5e-3, 2e-3, 1e-3, 5e-4]
    for gd_model in gd_models:
        results[gd_model] = {}
        for lr in lrs:
            results[gd_model][lr] = np.zeros(5)
            for id in range(5):
                elbo_history = torch.load(f'model/madelon_gd_d={d}_lr={lr}_method={gd_model}_id={id}.pt')['elbo_history']
                results[gd_model][lr][id] = find_index(elbo_history, threshold)
    for ngd_model in ngd_models:
        results[ngd_model] = {}
        for lr in lrs:
            results[ngd_model][lr] = np.zeros(5)
            for id in range(5):
                elbo_history = torch.load(f'model/madelon_ngd_proj={ngd_model}_d={d}_lr={lr}_id={id}.pt')['elbo_history']
                results[ngd_model][lr][id] = find_index(elbo_history, threshold)
    
    for model, lr_dict in results.items():
        lrs = sorted(lr_dict.keys())
        medians = []
        p25 = []
        p75 = []

        for lr in lrs:
            data = lr_dict[lr]
            medians.append(np.median(data))
            p25.append(np.percentile(data, 25))
            p75.append(np.percentile(data, 75))
            """if(model in ngd_models and lr > 0.09):
                print(model, lr, np.median(data), np.percentile(data, 25), np.percentile(data, 75))"""
            
        medians = np.array(medians)
        p25 = np.array(p25)
        p75 = np.array(p75)

        if mode == 0:
            if model == 'proj':
                plt.plot(lrs, medians, label=f'Proj-SGD', marker='^')
            elif model == 'prox':
                plt.plot(lrs, medians, label=f'Prox-SGD', marker='v')
            elif model == "None":
                plt.plot(lrs, medians, label=f'SNGD', marker='x')
            else:
                plt.plot(lrs, medians, label=f'Proj-SNGD', marker='o')
        elif mode == 1:
            if model == "None":
                plt.plot(lrs, medians, label=rf'U=$\infty$, D=$\infty$', marker='x')
            else:
                if model[0] == 3:
                    plt.plot(lrs, medians, label=f'U={model[0]}, D={model[1]}', marker='o')
                elif model[0] == 2:
                    plt.plot(lrs, medians, label=f'U={model[0]}, D={model[1]}', marker='s')
                elif model[0] == 1.5:
                    plt.plot(lrs, medians, label=f'U={model[0]}, D={model[1]}', marker='*')
        plt.fill_between(lrs, p25, p75, alpha=0.3)
            
    plt.xscale('log')
    plt.xlabel(r'$\gamma_0$')
    plt.ylabel('Number of Iterations')
    plt.legend(loc='upper right')
    plt.grid(True)
    plt.tight_layout()
    plt.savefig(f'figure/madelon robust {mode+1}.pdf')
    plt.clf()

experiment_madelon()
for mode in range(2):
    if mode == 1:
        custom_colors = ['#5B478B', '#356792', '#59A4A4', '#3C8358']
        plt.rcParams.update({'axes.prop_cycle': plt.cycler('color', custom_colors)})
    plot_results_convergence(mode)
    plot_results_robustness(mode)
