import os
import numpy as np
import torch
import torch.nn as nn
import torch.optim as optim
from torch.optim.lr_scheduler import StepLR
import pandas as pd
from concurrent.futures import ProcessPoolExecutor
import sys

# Add parent directory to path to import modules
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.abspath(__file__))))

from generate_lap import generate_lap_dataset
from LFRCov_frobenius import LFRCovFrobenius_torch

# Parameters (matching nn_loc_cov_network.py)
p = 4
n = 1250
q = 10
batch_size = 64
n_epochs = 1000
hidden_dim = 64
lr = 0.5
patience = 5
delta = 1e-4  # Minimum improvement threshold for early stopping
lambda_reg = 0.01  # Regularization hyperparameter for 1/h term

def set_torch_threads():
    torch.set_num_threads(1)

def frobenius_distance(S1, S2):
    """
    Frobenius distance between two matrices:
    d(S1,S2) = ||S1 - S2||_F
    """
    # Ensure matrices are symmetric
    S1 = 0.5 * (S1 + S1.transpose(-1, -2))
    S2 = 0.5 * (S2 + S2.transpose(-1, -2))
    
    # Calculate Frobenius norm of the difference
    return torch.norm(S1 - S2, p='fro')

def frechet_variance_frobenius(M_list):
    """Compute Fréchet variance using Frobenius distance"""
    n = len(M_list)
    if n <= 1:
        return torch.tensor(0.0, device=M_list[0].device)
    
    # Compute Fréchet mean using Frobenius metric (arithmetic mean)
    M_mean = compute_frechet_mean_frobenius(M_list)
    
    # Compute distances to mean
    dists = torch.stack([frobenius_distance(M, M_mean) for M in M_list])
    
    return dists.mean()

def compute_frechet_mean_frobenius(M_list):
    """Compute Fréchet mean using Frobenius metric (arithmetic mean)"""
    n = len(M_list)
    if n == 1:
        return M_list[0]
    
    # Under Frobenius metric, Fréchet mean is just arithmetic mean
    M_stack = torch.stack(M_list, dim=0)
    M_mean = M_stack.mean(dim=0)
    
    return M_mean

def lfr_loss_selffit(X, M_list, theta, h):
    """LFR loss with self-fitting (no separate validation)"""
    device = X.device
    n_samples = X.shape[0]
    
    # Compute Z = <X_i, theta_i> for each sample (theta is batch-aligned)
    Z = (X * theta).sum(dim=1, keepdim=True)
    
    # M_list is already a list of n matrices (n, q, q)
    # Use LFR with Frobenius metric
    result = LFRCovFrobenius_torch(
        x=Z, M=M_list, xout=Z, h=h, 
        kernel='gauss', dtype=torch.float64, device=device
    )
    
    M_hat_list = result['Mout']
    
    # Compute Frobenius distances
    distances = []
    for i in range(n_samples):
        dist = frobenius_distance(M_list[i], M_hat_list[i])
        distances.append(dist)
    
    return torch.stack(distances).mean()

class ThetaMLP(nn.Module):
    def __init__(self, input_dim, hidden_dim=64, dropout_rate=0.3):
        super(ThetaMLP, self).__init__()
        self.fc1 = nn.Linear(input_dim, hidden_dim)
        self.ln1 = nn.LayerNorm(hidden_dim)
        self.act1 = nn.LeakyReLU()
        self.drop1 = nn.Dropout(dropout_rate)
        
        self.fc2 = nn.Linear(hidden_dim, hidden_dim)
        self.ln2 = nn.LayerNorm(hidden_dim)
        self.act2 = nn.LeakyReLU()
        self.drop2 = nn.Dropout(dropout_rate)
        
        self.fc3 = nn.Linear(hidden_dim, input_dim)
        nn.init.kaiming_normal_(self.fc1.weight, nonlinearity='leaky_relu')
        nn.init.kaiming_normal_(self.fc2.weight, nonlinearity='leaky_relu')
        nn.init.kaiming_normal_(self.fc3.weight, nonlinearity='leaky_relu')
        for layer in (self.fc1, self.fc2, self.fc3):
            if layer.bias is not None:
                nn.init.zeros_(layer.bias)
        
    def forward(self, x):
        x = self.fc1(x)
        x = self.ln1(x)
        x = self.act1(x)
        x = self.drop1(x)
        
        x = self.fc2(x)
        x = self.ln2(x)
        x = self.act2(x)
        x = self.drop2(x)
        
        x = self.fc3(x)
        
        # Normalize to unit vector and fix sign ambiguity
        norm = torch.norm(x, dim=1, keepdim=True)
        theta = x / (norm + 1e-8)
        sign = torch.where(theta[:, :1] < 0, -1.0, 1.0)
        theta = theta * sign
        
        return theta

def run_single_sim(seed):
    set_torch_threads()
    torch.manual_seed(seed)
    np.random.seed(seed)
    
    device = torch.device('cpu')  # Use CPU for parallel processing
    
    # Generate data using generate_lap_dataset
    X, M_list, conditional_means = generate_lap_dataset(
        n=n, m=1000, q=q, random_seed=seed, device=device, dtype=torch.float32
    )
    X = X.to(device)
    M_list = [M.to(device) for M in M_list]
    conditional_means = [M.to(device) for M in conditional_means]
    
    # Split data
    idx = np.arange(n)
    np.random.shuffle(idx)
    n_train = int(0.4 * n)
    n_val = int(0.4 * n)
    n_test = 200
    
    idx_train = idx[:n_train]
    idx_val = idx[n_train:n_train+n_val]
    idx_test = idx[n_train+n_val:n_train+n_val+n_test]
    
    X_train, X_val, X_test = X[idx_train], X[idx_val], X[idx_test]
    M_train = [M_list[i] for i in idx_train]
    M_val = [M_list[i] for i in idx_val]
    M_test = [M_list[i] for i in idx_test]
    conditional_means_test = [conditional_means[i] for i in idx_test]
    
    # Normalize predictors using training statistics
    X_mean = X_train.mean(dim=0, keepdim=True)
    X_std = X_train.std(dim=0, keepdim=True).clamp_min(1e-8)
    X_train = (X_train - X_mean) / X_std
    X_val = (X_val - X_mean) / X_std
    X_test = (X_test - X_mean) / X_std
    
    # Compute global Fréchet variance for normalization
    with torch.no_grad():
        frechet_var = frechet_variance_frobenius(M_train)
        frechet_var = frechet_var + 1e-8
    
    # Create model
    model = ThetaMLP(p, hidden_dim, dropout_rate=0.2).to(device)
    # Learnable bandwidth parameter, clamped to [0.1, 1.0]
    h_param = nn.Parameter(torch.tensor(0.5, device=device))
    optimizer = optim.Adam(list(model.parameters()) + [h_param], lr=lr, weight_decay=1e-3)
    scheduler = StepLR(optimizer, step_size=100, gamma=0.5)
    
    best_val_loss = float('inf')
    patience_counter = 0
    best_model_state = None
    best_h = None
    
    train_dataset = torch.utils.data.TensorDataset(X_train, torch.stack(M_train))
    train_loader = torch.utils.data.DataLoader(train_dataset, batch_size=batch_size, shuffle=True)
    
    val_losses = []
    train_losses = []
    
    for epoch in range(n_epochs):
        model.train()
        total_loss = 0.0
        total_samples = 0
        
        for X_batch, M_batch in train_loader:
            optimizer.zero_grad()
            
            theta_batch = model(X_batch)
            h_value = torch.clamp(h_param, min=0.1, max=1.0)
            
            # Convert M_batch from tensor to list
            M_batch_list = [M_batch[j] for j in range(M_batch.shape[0])]
            
            lfr = lfr_loss_selffit(X_batch, M_batch_list, theta_batch, h_value)
            norm_loss = lfr / frechet_var
            loss = norm_loss + (1.0 / h_value) * lambda_reg
            
            loss.backward()
            torch.nn.utils.clip_grad_norm_(model.parameters(), max_norm=1.0)
            optimizer.step()
            
            total_loss += loss.item() * X_batch.size(0)
            total_samples += X_batch.size(0)
        
        avg_train_loss = total_loss / total_samples
        train_losses.append(avg_train_loss)
        scheduler.step()
        
        # Print progress
        if (epoch + 1) % 50 == 0 or epoch == 0:
            print(f"[Seed {seed}] Epoch {epoch+1}/{n_epochs}, Loss: {avg_train_loss:.6f}")
        
        # Validation
        model.eval()
        with torch.no_grad():
            theta_train = model(X_train)
            theta_val = model(X_val)
            h_eval = torch.clamp(h_param.detach(), min=0.1, max=1.0)
            
            val_lfr = lfr_loss_selffit(X_val, M_val, theta_val, h_eval)
            val_loss = (val_lfr / frechet_var + (1.0 / h_eval) * lambda_reg).item()
            val_losses.append(val_loss)
            
            # Standard early stopping criterion:
            # Stop when val_loss(t) > min val_loss(s) - δ for 'patience' consecutive epochs
            if val_loss < best_val_loss - delta:
                best_val_loss = val_loss
                patience_counter = 0
                best_model_state = {k: v.detach().cpu().clone() for k, v in model.state_dict().items()}
                best_h = h_param.detach().cpu().clone()
            else:
                patience_counter += 1
                if patience_counter >= patience:
                    if best_model_state is not None:
                        model.load_state_dict(best_model_state)
                        h_param.data = best_h.to(device)
                    break
    
    # Compute Fréchet mean theta on training set
    with torch.no_grad():
        model.eval()
        theta_final = model(X_train)
        theta_mean = theta_final.mean(dim=0)  # Column mean
        theta_mean_np = theta_mean.detach().cpu().numpy()
        
        # Also compute theta predictions for X_test
        theta_test = model(X_test)
        theta_test_np = theta_test.detach().cpu().numpy()
        
        # Predict matrices for test set
        h_final = torch.clamp(h_param.detach(), min=0.1, max=1.0)
        M_pred_test = []
        
        for i in range(len(X_test)):
            Z_test = (X_test[i:i+1] * theta_test[i:i+1]).sum(dim=1, keepdim=True)
            Z_train = (X_train * theta_final).sum(dim=1, keepdim=True)
            
            result_test = LFRCovFrobenius_torch(
                x=Z_train, M=M_train, xout=Z_test, h=h_final,
                kernel='gauss', dtype=torch.float64, device=device
            )
            M_pred_test.append(result_test['Mout'][0])
        
        # Calculate average Frobenius distance between predicted and true conditional means
        frob_distances = []
        for i in range(len(M_test)):
            dist = frobenius_distance(M_pred_test[i], conditional_means_test[i])
            frob_distances.append(dist.item())
        
        avg_frob_distance = np.mean(frob_distances)
        print(f"[Seed {seed}] Average Frobenius distance between predicted and true conditional means (test set): {avg_frob_distance:.6f}")
    
    return theta_mean_np, theta_test_np, avg_frob_distance

if __name__ == '__main__':
    num_cpus = os.cpu_count() - 1
    num_simulations = 200
    print(f"Running {num_simulations} simulations using {num_cpus} CPUs...")
    
    results = []
    failed_seeds = []
    
    with ProcessPoolExecutor(max_workers=num_cpus) as executor:
        # Submit all tasks
        future_to_seed = {executor.submit(run_single_sim, seed): seed for seed in range(1, num_simulations + 1)}
        
        # Process results as they complete
        for future in future_to_seed:
            seed = future_to_seed[future]
            try:
                result = future.result()
                results.append(result)
            except Exception as e:
                failed_seeds.append(seed)
                print(f"\n❌ ERROR: Seed {seed} failed with error:")
                print(f"   {type(e).__name__}: {str(e)}")
                import traceback
                traceback.print_exc()
    
    if failed_seeds:
        print(f"\n⚠️  Failed seeds: {failed_seeds}")
        print(f"   Successful simulations: {len(results)}/{num_simulations}")
    
    if len(results) == 0:
        print("\n❌ All simulations failed! Exiting.")
        exit(1)
    
    all_thetas, all_theta_tests, all_frob_means = zip(*results)
    all_thetas = np.stack(all_thetas)
    all_theta_tests = np.stack(all_theta_tests)
    all_frob_means = np.array(all_frob_means)
    
    # Save Fréchet mean thetas
    pd.DataFrame(all_thetas, columns=[f"theta_{i+1}" for i in range(p)]).to_csv("frechet_mean_thetas500.csv", index=False)
    
    # Save X_test theta predictions (flattened)
    n_test = all_theta_tests.shape[1]
    n_successful = len(results)
    theta_test_flat = all_theta_tests.reshape(n_successful, n_test * p)
    theta_test_columns = [f"theta_test_{i+1}_{j+1}" for i in range(n_test) for j in range(p)]
    pd.DataFrame(theta_test_flat, columns=theta_test_columns).to_csv("x_test_theta_predictions500.csv", index=False)
    
    # Save mean Frobenius distances
    pd.DataFrame(all_frob_means, columns=["mean_frobenius_distance"]).to_csv("mean_frobenius_distances500.csv", index=False)
    
    print(f"\n✅ Saved {n_successful} Fréchet mean thetas to frechet_mean_thetas500.csv")
    print(f"✅ Saved X_test theta predictions to x_test_theta_predictions500.csv")
    print(f"✅ Saved mean Frobenius distances to mean_frobenius_distances500.csv")
    if failed_seeds:
        print(f"\n⚠️  Note: {len(failed_seeds)} simulation(s) failed (seeds: {failed_seeds})")
