import os
import logging
import pickle
import torch
import torch.nn as nn
import numpy as np
import sklearn as sk
import random
import tqdm as tqdm
from torch.nn.utils.rnn import pad_sequence
import matplotlib.pyplot as plt
from sklearn.manifold import TSNE


def get_data_split(base_path='./data/P12', split_path='', dataset='P12'):
    # load data
    if dataset == 'mimic3':
        Ptrain = np.load(base_path + '/mimic3_train_x.npy', allow_pickle=True)
        Pval = np.load(base_path + '/mimic3_val_x.npy', allow_pickle=True)
        Ptest = np.load(base_path + '/mimic3_test_x.npy', allow_pickle=True)
        ytrain = np.load(base_path + '/mimic3_train_y.npy', allow_pickle=True).reshape(-1, 1)
        yval = np.load(base_path + '/mimic3_val_y.npy', allow_pickle=True).reshape(-1, 1)
        ytest = np.load(base_path + '/mimic3_test_y.npy', allow_pickle=True).reshape(-1, 1)
        
        return Ptrain, Pval, Ptest, ytrain, yval, ytest

    if dataset == 'P12' or dataset == 'physionet':
        Pdict_list = np.load(base_path + '/processed_data/PTdict_list.npy', allow_pickle=True)
        arr_outcomes = np.load(base_path + '/processed_data/arr_outcomes.npy', allow_pickle=True)
    elif dataset == 'P19':
        Pdict_list = np.load(base_path + '/processed_data/PT_dict_list_6.npy', allow_pickle=True)
        arr_outcomes = np.load(base_path + '/processed_data/arr_outcomes_6.npy', allow_pickle=True)
    elif dataset == 'PAM':
        Pdict_list = np.load(base_path + '/processed_data/PTdict_list.npy', allow_pickle=True)
        arr_outcomes = np.load(base_path + '/processed_data/arr_outcomes.npy', allow_pickle=True)

    idx_train, idx_val, idx_test = np.load(base_path + split_path, allow_pickle=True)
    
    # extract train/val/test examples
    Ptrain = Pdict_list[idx_train]
    Pval = Pdict_list[idx_val]
    Ptest = Pdict_list[idx_test]
    y = arr_outcomes[:, -1].reshape((-1, 1))
    ytrain = y[idx_train]
    yval = y[idx_val]
    ytest = y[idx_test]
    
    return Ptrain, Pval, Ptest, ytrain, yval, ytest

def getStats(P_tensor):
    N, T, F = P_tensor.shape
    Pf = P_tensor.transpose((2, 0, 1)).reshape(F, -1)
    mf = np.zeros((F, 1))
    stdf = np.ones((F, 1))
    eps = 1e-7
    for f in range(F):
        vals_f = Pf[f, :]
        vals_f = vals_f[vals_f > 0]
        if vals_f.size == 0:
            mf[f] = 0.0
        else:
            mf[f] = np.mean(vals_f)
        stdf[f] = np.std(vals_f)
        # stdf[f] = np.max([stdf[f], eps])
    return mf, stdf

def getStats_static(P_tensor, dataset='P12'):
    N, S = P_tensor.shape
    Ps = P_tensor.transpose((1, 0))
    ms = np.zeros((S, 1))
    ss = np.ones((S, 1))

    if dataset == 'P12' or dataset == 'physionet':
        # ['Age' 'Gender=0' 'Gender=1' 'Height' 'ICUType=1' 'ICUType=2' 'ICUType=3' 'ICUType=4' 'Weight']
        bool_categorical = [0, 1, 1, 0, 1, 1, 1, 1, 0]
    elif dataset == 'P19':
        # ['Age' 'Gender' 'Unit1' 'Unit2' 'HospAdmTime' 'ICULOS']
        bool_categorical = [0, 1, 0, 0, 0, 0]

    for s in range(S):
        if bool_categorical[s] == 0:  # if not categorical
            vals_s = Ps[s, :]
            vals_s = vals_s[vals_s > 0]
            ms[s] = np.mean(vals_s)
            ss[s] = np.std(vals_s)
    return ms, ss

def mask_normalize(P_tensor, mf, stdf):
    """ Normalize time series variables. Missing ones are set to zero after normalization. """
    N, T, F = P_tensor.shape
    Pf = P_tensor.transpose((2, 0, 1)).reshape(F, -1)
    M = 1 * (P_tensor > 0) + 0 * (P_tensor <= 0)
    M_3D = M.transpose((2, 0, 1)).reshape(F, -1)
    for f in range(F):
        Pf[f] = (Pf[f] - mf[f]) / (stdf[f] + 1e-18)
    Pf = Pf * M_3D
    Pnorm_tensor = Pf.reshape((F, N, T)).transpose((1, 2, 0))
    Pfinal_tensor = np.concatenate([Pnorm_tensor, M], axis=2)
    return Pfinal_tensor

def mask_normalize_static(P_tensor, ms, ss):
    N, S = P_tensor.shape
    Ps = P_tensor.transpose((1, 0))

    # input normalization
    for s in range(S):
        Ps[s] = (Ps[s] - ms[s]) / (ss[s] + 1e-18)

    # set missing values to zero after normalization
    for s in range(S):
        idx_missing = np.where(Ps[s, :] <= 0)
        Ps[s, idx_missing] = 0

    # reshape back
    Pnorm_tensor = Ps.reshape((S, N)).transpose((1, 0))
    return Pnorm_tensor


def tensorize_normalize_extract_feature_pam_patch(P, y, mf, stdf, args):
    T, F = P[0].shape
    P_time = np.zeros((len(P), T, 1))
    for i in range(len(P)):
        tim = torch.linspace(0, T, T).reshape(-1, 1)
        P_time[i] = tim
    P_tensor = mask_normalize(P, mf, stdf)
    P_tensor = torch.Tensor(P_tensor)

    P_time = torch.Tensor(P_time) / 60.0

    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    combined_vals = []
    combined_mask = []
    combined_tt = []
    for b in range(P_tensor.shape[0]):
        record = P_tensor[b]
        X_list = []
        Mask_list = []
        T_list = []

        cur_x = record[:, :F]
        cur_mask = record[:, F: 2 * F]
        cur_t = P_time[b][:, 0]
        if cur_t[0] == 0.0:
            cur_t[0] = 0.0001

        if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
            st, ed = 0, args.patch_size / 48
        else:
            st, ed = 0, args.patch_size / 60
  
        npatch = args.npatch
        for i in range(npatch):
            cur_ind = np.where((cur_t > st) & (cur_t <= ed))[0]
            if len(cur_ind) == 0:
                X_list.append(torch.zeros(size=[1, F]))
                Mask_list.append(torch.zeros(size=[1, F]))
                T_list.append(torch.zeros(size=[1]))
            else:
                X_list.append(torch.tensor(cur_x[cur_ind]))
                Mask_list.append(torch.tensor(cur_mask[cur_ind]))
                T_list.append(torch.tensor(cur_t[cur_ind]))
                
            if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
                st += args.stride / 48
                ed += args.stride / 48
            else:
                st += args.stride / 60
                ed += args.stride / 60

        combined_vals.append(pad_sequence(X_list).float())
        combined_mask.append(pad_sequence(Mask_list).float())
        combined_tt.append(pad_sequence(T_list).float())

    observed_data = pad_sequence(combined_vals, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)
    observed_tp = pad_sequence(combined_tt, batch_first=True).unsqueeze(-1).repeat(1, 1, 1, F).permute(0, 2, 1,3)  # (T_o, )
    observed_mask = pad_sequence(combined_mask, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)

    return observed_data, observed_mask, None, observed_tp, y_tensor, observed_data.shape[2]


def tensorize_normalize_extract_feature_pam(P, y, mf, stdf, args):
    T, F = P[0].shape
    print(P[0].shape)
    P_time = np.zeros((len(P), T, 1))
    for i in range(len(P)):
        tim = torch.linspace(0, T, T).reshape(-1, 1)
        P_time[i] = tim
    P_tensor = mask_normalize(P, mf, stdf)
    P_tensor = torch.Tensor(P_tensor)

    P_time = torch.Tensor(P_time) / 60.0

    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    observed_data = torch.FloatTensor(P_tensor[:, :, :F])
    observed_mask = torch.FloatTensor(P_tensor[:, :, F:2*F])
    observed_tp = torch.FloatTensor(P_time) 

    return observed_data, observed_mask, None, observed_tp, y_tensor, observed_data.shape[1]


def tensorize_normalize_extract_feature_patch(P, y, mf, stdf, ms, ss, args):
    T, F = P[0]['arr'].shape
    D = len(P[0]['extended_static'])

    P_tensor = np.zeros((len(P), T, F))
    P_time_tensor = np.zeros((len(P), T, F))
    P_mask_tensor = np.zeros((len(P), T, F))
    P_static_tensor = np.zeros((len(P), D))
    for i in range(len(P)):
        P_tensor[i] = P[i]['arr']
        P[i]['time'] = P[i]['time'] / 60 if T == 60 else P[i]['time'] / 2880
        P_time_tensor[i] = P[i]['time']
        P_mask_tensor[i] = P[i]['arr'] > 0
        P_static_tensor[i] = P[i]['extended_static']

    P_tensor = mask_normalize(P_tensor, mf, stdf)
    P_static_tensor = mask_normalize_static(P_static_tensor, ms, ss)

    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    combined_vals = []
    combined_mask = []
    combined_tt = []
    for b in range(P_tensor.shape[0]):
        record = P_tensor[b]
        X_list = []
        Mask_list = []
        T_list = []

        cur_x = record[:, :F]
        cur_mask = record[:, F: 2 * F]
        cur_t = P_time_tensor[b][:, 0]
        if cur_t[0] == 0.0:
            cur_t[0] = 0.0001

        if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
            st, ed = 0, args.patch_size / 48
        else:
            st, ed = 0, args.patch_size / 60

        npatch = args.npatch
        for i in range(npatch):
            cur_ind = np.where((cur_t > st) & (cur_t <= ed))[0]
            if len(cur_ind) == 0:
                X_list.append(torch.zeros(size=[1, F]))
                Mask_list.append(torch.zeros(size=[1, F]))
                T_list.append(torch.zeros(size=[1]))
            else:
                X_list.append(torch.tensor(cur_x[cur_ind]))
                Mask_list.append(torch.tensor(cur_mask[cur_ind]))
                T_list.append(torch.tensor(cur_t[cur_ind]))
   
            if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
                st += args.stride / 48
                ed += args.stride / 48
            else:
                st += args.stride / 60
                ed += args.stride / 60
        
        combined_vals.append(pad_sequence(X_list).float())
        combined_mask.append(pad_sequence(Mask_list).float())
        combined_tt.append(pad_sequence(T_list).float())

    observed_data = pad_sequence(combined_vals, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)
    observed_tp = pad_sequence(combined_tt, batch_first=True).unsqueeze(-1).repeat(1, 1, 1, F).permute(0, 2, 1,3)  # (T_o, )
    observed_mask = pad_sequence(combined_mask, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)

    return observed_data, observed_mask, torch.FloatTensor(P_static_tensor), observed_tp, y_tensor, observed_data.shape[2]


def tensorize_normalize_extract_feature_patchtst(P, y, mf, stdf, ms, ss, args):
    T, F = P[0]['arr'].shape
    D = len(P[0]['extended_static'])

    P_tensor = np.zeros((len(P), T, F))
    P_time_tensor = np.zeros((len(P), T, F))
    P_mask_tensor = np.zeros((len(P), T, F))
    P_static_tensor = np.zeros((len(P), D))
    for i in range(len(P)):
        P_tensor[i] = P[i]['arr']
        P[i]['time'] = P[i]['time'] / 60 if T == 60 else P[i]['time'] / 2880
        P_time_tensor[i] = P[i]['time']
        P_mask_tensor[i] = P[i]['arr'] > 0
        P_static_tensor[i] = P[i]['extended_static']

    P_tensor = mask_normalize(P_tensor, mf, stdf)
    P_static_tensor = mask_normalize_static(P_static_tensor, ms, ss)

    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    combined_vals = []
    combined_mask = []
    combined_tt = []
    for b in range(P_tensor.shape[0]):
        record = P_tensor[b]
        X_list = []
        Mask_list = []
        T_list = []

        cur_x = record[:, :F]
        cur_mask = record[:, F: 2 * F]
        cur_t = P_time_tensor[b][:, 0]
        if cur_t[0] == 0.0:
            cur_t[0] = 0.0001

        st, ed = 0, args.patch_size / 48
        
        fixed_len = 52
        npatch = args.npatch
        for i in range(npatch):
            cur_ind = np.where((cur_t > st) & (cur_t <= ed))[0]
            if len(cur_ind) == 0:
                X_list.append(torch.zeros(size=[fixed_len, F]))
                Mask_list.append(torch.zeros(size=[fixed_len, F]))
                T_list.append(torch.zeros(size=[fixed_len]))
            else:
                x_p = torch.tensor(cur_x[cur_ind])
                m_p = torch.tensor(cur_mask[cur_ind])
                t_p = torch.tensor(cur_t[cur_ind])
                
                if x_p.size(0) > fixed_len:
                    x_p, m_p, t_p = x_p[:fixed_len], m_p[:fixed_len], t_p[:fixed_len]
                else:
                    diff = fixed_len - x_p.size(0)
                    x_p = torch.cat([x_p, torch.zeros(diff, F)], dim=0)
                    m_p = torch.cat([m_p, torch.zeros(diff, F)], dim=0)
                    t_p = torch.cat([t_p, torch.zeros(diff)], dim=0)
                    
                X_list.append(x_p)
                Mask_list.append(m_p)
                T_list.append(t_p)
    
            st += args.stride / 48
            ed += args.stride / 48
        
        combined_vals.append(pad_sequence(X_list).float())
        combined_mask.append(pad_sequence(Mask_list).float())
        combined_tt.append(pad_sequence(T_list).float())

    observed_data = pad_sequence(combined_vals, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)
    observed_tp = pad_sequence(combined_tt, batch_first=True).unsqueeze(-1).repeat(1, 1, 1, F).permute(0, 2, 1,3)  # (T_o, )
    observed_mask = pad_sequence(combined_mask, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)

    return observed_data, observed_mask, torch.FloatTensor(P_static_tensor), observed_tp, y_tensor, observed_data.shape[2]


def tensorize_normalize_extract_feature(P, y, mf, stdf, ms, ss, args):
    T, F = P[0]['arr'].shape
    D = len(P[0]['extended_static'])

    P_tensor = np.zeros((len(P), T, F))
    P_time_tensor = np.zeros((len(P), T, F))
    P_mask_tensor = np.zeros((len(P), T, F))
    P_static_tensor = np.zeros((len(P), D))
    for i in range(len(P)):
        P_tensor[i] = P[i]['arr']
        P[i]['time'] = P[i]['time'] / 60 if T == 60 else P[i]['time'] / 2880
        P_time_tensor[i] = P[i]['time']
        P_mask_tensor[i] = P[i]['arr'] > 0
        P_static_tensor[i] = P[i]['extended_static']

    P_tensor = mask_normalize(P_tensor, mf, stdf)
    P_static_tensor = mask_normalize_static(P_static_tensor, ms, ss)

    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    observed_data = torch.FloatTensor(P_tensor[:, :, :F])
    observed_mask = torch.FloatTensor(P_tensor[:, :, F:2*F])
    observed_tp = torch.FloatTensor(P_time_tensor) 
    static_tensor = torch.FloatTensor(P_static_tensor)

    return observed_data, observed_mask, static_tensor, observed_tp, y_tensor, observed_data.shape[1]


def tensorize_normalize_exact_feature_mimic3_patch(P, y, mf, stdf, args):
    T, F = 292, 16

    P_tensor = np.zeros((len(P), T, F))
    P_time_tensor = np.zeros((len(P), T, F))
    P_mask_tensor = np.zeros((len(P), T, F))

    for i in range(len(P)):
        P_tensor[i][:P[i][4]] = P[i][2]
        P[i][1] = P[i][1] / 48

        P_time_tensor[i][:P[i][4]] = np.tile(P[i][1], (F, 1)).T
        P_mask_tensor[i][: P[i][4]] = P[i][3]

    P_tensor = mask_normalize(P_tensor, mf, stdf)
    y_tensor = y
    y_tensor = torch.Tensor(y_tensor[:, 0]).type(torch.LongTensor)

    combined_vals = []
    combined_mask = []
    combined_tt = []
    # for b in range(100):
    for b in range(P_tensor.shape[0]):
        record = P_tensor[b]
        X_list = []
        Mask_list = []
        T_list = []

        cur_x = record[:, :F]
        cur_mask = record[:, F: 2 * F]
        cur_t = P_time_tensor[b][:, 0]
        if cur_t[0] == 0.0:
            cur_t[0] = 0.0001

        if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
            st, ed = 0, args.patch_size / 48
        else:
            st, ed = 0, args.patch_size / 60

        npatch = args.npatch
        for i in range(npatch):
            cur_ind = np.where((cur_t > st) & (cur_t <= ed))[0]
            if len(cur_ind) == 0:
                X_list.append(torch.zeros(size=[1, F]))
                Mask_list.append(torch.zeros(size=[1, F]))
                T_list.append(torch.zeros(size=[1]))
            else:
                X_list.append(torch.tensor(cur_x[cur_ind]))
                Mask_list.append(torch.tensor(cur_mask[cur_ind]))
                T_list.append(torch.tensor(cur_t[cur_ind]))

            if args.dataset == 'physionet' or args.dataset == 'P12' or args.dataset == 'mimic3':
                st += args.stride / 48
                ed += args.stride / 48
            else:
                st += args.stride / 60
                ed += args.stride / 60

        combined_vals.append(pad_sequence(X_list).float())
        combined_mask.append(pad_sequence(Mask_list).float())
        combined_tt.append(pad_sequence(T_list).float())

    observed_data = pad_sequence(combined_vals, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)
    observed_tp = pad_sequence(combined_tt, batch_first=True).unsqueeze(-1).repeat(1, 1, 1, F).permute(0, 2, 1,
                                                                                                       3)  # (T_o, )
    observed_mask = pad_sequence(combined_mask, batch_first=True).permute(0, 2, 1, 3)  # (n_batch, T_o, D)

    return observed_data, observed_mask, None, observed_tp, y_tensor, observed_data.shape[1]*observed_data.shape[2]


def evaluate_model_patch(model, P_tensor, P_mask_tensor, P_static_tensor, P_time_tensor,
                      batch_size=100, n_classes=2, static=None):
    model.eval()
    P_tensor = P_tensor.cuda()
    P_mask_tensor = P_mask_tensor.cuda()
    P_time_tensor = P_time_tensor.cuda()

    if P_static_tensor is None:
        Pstatic = None
    else:
        P_static_tensor = P_static_tensor.cuda()
        N, Fs = P_static_tensor.shape
    
    if len(P_tensor.shape) == 4:
        N, _, F, Ff = P_tensor.shape
    else:
        N, F, Ff = P_tensor.shape
     
    n_batches, rem = N // batch_size, N % batch_size
    out = torch.zeros(N, n_classes)
    start = 0
    for i in range(n_batches):
        P = P_tensor[start:start + batch_size]
        P_mask = P_mask_tensor[start:start + batch_size]
        P_time = P_time_tensor[start:start + batch_size]

        if P_static_tensor is not None:
            Pstatic = P_static_tensor[start:start + batch_size]
        middleoutput = model.classification(P, P_time, P_mask, Pstatic)
        out[start:start + batch_size] = middleoutput.detach().cpu()
        start += batch_size
    if rem > 0:
        P = P_tensor[start:start + rem]
        P_mask = P_mask_tensor[start:start + batch_size]
        P_time = P_time_tensor[start:start + rem]

        if P_static_tensor is not None:
            Pstatic = P_static_tensor[start:start + rem]
        whatever = model.classification(P, P_time, P_mask, Pstatic)
        out[start:start + rem] = whatever.detach().cpu()
    return out


def t_sne_visualize(model, y_test, P_tensor, P_mask_tensor, P_static_tensor, P_time_tensor, 
                    batch_size=100, save_path='./logs/tsne_plot.png'):
    model.eval()
    
    # 1. 고유 라벨 확인 (y_test의 클래스 개수 파악)
    unique_labels = np.unique(y_test)
    num_classes = len(unique_labels)
    print(f"Detected {num_classes} classes: {unique_labels}")

    # 데이터 GPU 이동
    P_tensor = P_tensor.cuda()
    P_mask_tensor = P_mask_tensor.cuda()
    P_time_tensor = P_time_tensor.cuda()
    if P_static_tensor is not None:
        P_static_tensor = P_static_tensor.cuda()

    N = P_tensor.shape[0]
    n_batches, rem = N // batch_size, N % batch_size
    
    all_features = [] 
    start = 0

    with torch.no_grad():
        # 2. 잠재 벡터(h) 추출 루프
        for _ in range(n_batches):
            P = P_tensor[start:start + batch_size]
            P_mask = P_mask_tensor[start:start + batch_size]
            P_time = P_time_tensor[start:start + batch_size]
            Pstatic = P_static_tensor[start:start + batch_size] if P_static_tensor is not None else None
            
            h = model.classification(P, P_time, P_mask, Pstatic, feature=True)
            all_features.append(h.detach().cpu().numpy())
            start += batch_size

        if rem > 0:
            P = P_tensor[start:start + rem]
            P_mask = P_mask_tensor[start:start + rem]
            P_time = P_time_tensor[start:start + rem]
            Pstatic = P_static_tensor[start:start + rem] if P_static_tensor is not None else None
            
            h = model.classification(P, P_time, P_mask, Pstatic, feature=True)
            all_features.append(h.detach().cpu().numpy())

    # 3. 데이터 통합 및 t-SNE 실행
    features = np.concatenate(all_features, axis=0)
    features = features.reshape(features.shape[0], -1)
    
    print(f"Running t-SNE for {features.shape[0]} samples...")
    tsne = TSNE(n_components=2, perplexity=30, n_iter=1000, random_state=42)
    tsne_results = tsne.fit_transform(features)

    # 4. 시각화 (Label 개수에 맞춰 자동 색상 할당)
    plt.figure(figsize=(12, 8))
    
    # 클래스 개수에 최적화된 컬러맵 선택
    cmap = plt.cm.get_cmap('tab10' if num_classes <= 10 else 'Set3')
    
    for idx, label in enumerate(unique_labels):
        # 해당 클래스에 해당하는 포인트들만 필터링하여 시각화
        mask = (y_test.ravel() == label)
        plt.scatter(tsne_results[mask, 0], tsne_results[mask, 1], label=f'Class {int(label)}', color=cmap(idx), alpha=0.6, s=20)
    
    plt.legend(loc='best', title="Labels")
    plt.title(f't-SNE Visualization ({num_classes} Classes)')
    plt.xlabel('t-SNE 1')
    plt.ylabel('t-SNE 2')
    plt.grid(True, linestyle='--', alpha=0.3)

    # 파일 저장
    plt.savefig(save_path, dpi=300, bbox_inches='tight')
    plt.close()
    print(f"t-SNE plot saved: {save_path}")


def setup_seed(seed):
	torch.manual_seed(seed)
	torch.cuda.manual_seed(seed)
	torch.cuda.manual_seed_all(seed)
	np.random.seed(seed)
	random.seed(seed)
	torch.backends.cudnn.deterministic = True
    

def makedirs(dirname):
	if not os.path.exists(dirname):
		os.makedirs(dirname)


def save_checkpoint(state, save, epoch):
	if not os.path.exists(save):
		os.makedirs(save)
	filename = os.path.join(save, 'checkpt-%04d.pth' % epoch)
	torch.save(state, filename)

	
def get_logger(logpath, filepath, package_files=[],
			   displaying=True, saving=True, debug=False, mode='a'):
	logger = logging.getLogger()
	if debug:
		level = logging.DEBUG
	else:
		level = logging.INFO
	logger.setLevel(level)
	if saving:
		info_file_handler = logging.FileHandler(logpath, mode=mode)
		info_file_handler.setLevel(level)
		logger.addHandler(info_file_handler)
	if displaying:
		console_handler = logging.StreamHandler()
		console_handler.setLevel(level)
		logger.addHandler(console_handler)
	logger.info(filepath)

	for f in package_files:
		logger.info(f)
		with open(f, 'r') as package_f:
			logger.info(package_f.read())

	return logger


def inf_generator(iterable):
	"""Allows training with DataLoaders in a single infinite loop:
		for i, (x, y) in enumerate(inf_generator(train_loader)):
	"""
	iterator = iterable.__iter__()
	while True:
		try:
			yield iterator.__next__()
		except StopIteration:
			iterator = iterable.__iter__()

def dump_pickle(data, filename):
	with open(filename, 'wb') as pkl_file:
		pickle.dump(data, pkl_file)

def load_pickle(filename):
	with open(filename, 'rb') as pkl_file:
		filecontent = pickle.load(pkl_file)
	return filecontent

def make_dataset(dataset_type = "spiral",**kwargs):
	if dataset_type == "spiral":
		data_path = "data/spirals.pickle"
		dataset = load_pickle(data_path)["dataset"]
		chiralities = load_pickle(data_path)["chiralities"]
	elif dataset_type == "chiralspiral":
		data_path = "data/chiral-spirals.pickle"
		dataset = load_pickle(data_path)["dataset"]
		chiralities = load_pickle(data_path)["chiralities"]
	else:
		raise Exception("Unknown dataset type " + dataset_type)
	return dataset, chiralities


def split_last_dim(data):
	last_dim = data.size()[-1]
	last_dim = last_dim//2

	if len(data.size()) == 3:
		res = data[:,:,:last_dim], data[:,:,last_dim:]

	if len(data.size()) == 2:
		res = data[:,:last_dim], data[:,last_dim:]
	return res


def init_network_weights(net, std = 0.1):
	for m in net.modules():
		if isinstance(m, nn.Linear):
			nn.init.normal_(m.weight, mean=0, std=std)
			nn.init.constant_(m.bias, val=0)


def flatten(x, dim):
	return x.reshape(x.size()[:dim] + (-1, ))


def subsample_timepoints(data, time_steps, mask, n_tp_to_sample = None):
	# n_tp_to_sample: number of time points to subsample. If not None, sample exactly n_tp_to_sample points
	if n_tp_to_sample is None:
		return data, time_steps, mask
	n_tp_in_batch = len(time_steps)


	if n_tp_to_sample > 1:
		# Subsample exact number of points
		assert(n_tp_to_sample <= n_tp_in_batch)
		n_tp_to_sample = int(n_tp_to_sample)

		for i in range(data.size(0)):
			missing_idx = sorted(np.random.choice(np.arange(n_tp_in_batch), n_tp_in_batch - n_tp_to_sample, replace = False))

			data[i, missing_idx] = 0.
			if mask is not None:
				mask[i, missing_idx] = 0.
	
	elif (n_tp_to_sample <= 1) and (n_tp_to_sample > 0):
		# Subsample percentage of points from each time series
		percentage_tp_to_sample = n_tp_to_sample
		for i in range(data.size(0)):
			# take mask for current training sample and sum over all features -- figure out which time points don't have any measurements at all in this batch
			current_mask = mask[i].sum(-1).cpu()
			non_missing_tp = np.where(current_mask > 0)[0]
			n_tp_current = len(non_missing_tp)
			n_to_sample = int(n_tp_current * percentage_tp_to_sample)
			subsampled_idx = sorted(np.random.choice(non_missing_tp, n_to_sample, replace = False))
			tp_to_set_to_zero = np.setdiff1d(non_missing_tp, subsampled_idx)

			data[i, tp_to_set_to_zero] = 0.
			if mask is not None:
				mask[i, tp_to_set_to_zero] = 0.

	return data, time_steps, mask


def cut_out_timepoints(data, time_steps, mask, n_points_to_cut = None):
	# n_points_to_cut: number of consecutive time points to cut out
	if n_points_to_cut is None:
		return data, time_steps, mask
	n_tp_in_batch = len(time_steps)

	if n_points_to_cut < 1:
		raise Exception("Number of time points to cut out must be > 1")

	assert(n_points_to_cut <= n_tp_in_batch)
	n_points_to_cut = int(n_points_to_cut)

	for i in range(data.size(0)):
		start = np.random.choice(np.arange(5, n_tp_in_batch - n_points_to_cut-5), replace = False)

		data[i, start : (start + n_points_to_cut)] = 0.
		if mask is not None:
			mask[i, start : (start + n_points_to_cut)] = 0.

	return data, time_steps, mask

def get_device(tensor):
	device = torch.device("cuda:0")
	if tensor.is_cuda:
		device = tensor.get_device()
	return device

def sample_standard_gaussian(mu, sigma):
	device = get_device(mu)

	d = torch.distributions.normal.Normal(torch.Tensor([0.]).to(device), torch.Tensor([1.]).to(device))
	r = d.sample(mu.size()).squeeze(-1)
	return r * sigma.float() + mu.float()

def split_train_test(data, train_fraq = 0.8):
	n_samples = data.size(0)
	data_train = data[:int(n_samples * train_fraq)]
	data_test = data[int(n_samples * train_fraq):]
	return data_train, data_test

def split_train_test_data_and_time(data, time_steps, train_fraq = 0.8):
	n_samples = data.size(0)
	data_train = data[:int(n_samples * train_fraq)]
	data_test = data[int(n_samples * train_fraq):]

	assert(len(time_steps.size()) == 2)
	train_time_steps = time_steps[:, :int(n_samples * train_fraq)]
	test_time_steps = time_steps[:, int(n_samples * train_fraq):]

	return data_train, data_test, train_time_steps, test_time_steps

def get_next_batch(dataloader):
	# Make the union of all time points and perform normalization across the whole dataset
	data_dict = dataloader.__next__()
	
	return data_dict

def get_ckpt_model(ckpt_path, model, device):
	if not os.path.exists(ckpt_path):
		raise Exception("Checkpoint " + ckpt_path + " does not exist.")
	# Load checkpoint.
	checkpt = torch.load(ckpt_path)
	ckpt_args = checkpt['args']
	state_dict = checkpt['state_dict']
	model_dict = model.state_dict()

	# 1. filter out unnecessary keys
	state_dict = {k: v for k, v in state_dict.items() if k in model_dict}
	# 2. overwrite entries in the existing state dict
	model_dict.update(state_dict) 
	# 3. load the new state dict
	model.load_state_dict(state_dict)
	model.to(device)


def update_learning_rate(optimizer, decay_rate = 0.999, lowest = 1e-3):
	for param_group in optimizer.param_groups:
		lr = param_group['lr']
		lr = max(lr * decay_rate, lowest)
		param_group['lr'] = lr


def linspace_vector(start, end, n_points):
	# start is either one value or a vector
	size = np.prod(start.size())

	assert(start.size() == end.size())
	if size == 1:
		# start and end are 1d-tensors
		res = torch.linspace(start, end, n_points)
	else:
		# start and end are vectors
		res = torch.Tensor()
		for i in range(0, start.size(0)):
			res = torch.cat((res, 
				torch.linspace(start[i], end[i], n_points)),0)
		res = torch.t(res.reshape(start.size(0), n_points))
	return res

def reverse(tensor):
	idx = [i for i in range(tensor.size(0)-1, -1, -1)]
	return tensor[idx]


def create_net(n_inputs, n_outputs, n_layers = 1, 
	n_units = 100, nonlinear = nn.Tanh):
	layers = [nn.Linear(n_inputs, n_units)]
	for i in range(n_layers):
		layers.append(nonlinear())
		layers.append(nn.Linear(n_units, n_units))

	layers.append(nonlinear())
	layers.append(nn.Linear(n_units, n_outputs))
	return nn.Sequential(*layers)


def get_item_from_pickle(pickle_file, item_name):
	from_pickle = load_pickle(pickle_file)
	if item_name in from_pickle:
		return from_pickle[item_name]
	return None


def get_dict_template():
	return {"observed_data": None,
			"observed_tp": None,
			"data_to_predict": None,
			"tp_to_predict": None,
			"observed_mask": None,
			"mask_predicted_data": None,
			# "labels": None
			}


def normalize_data(data):
	reshaped = data.reshape(-1, data.size(-1))

	att_min = torch.min(reshaped, 0)[0]
	att_max = torch.max(reshaped, 0)[0]

	# we don't want to divide by zero
	att_max[ att_max == 0.] = 1.

	if (att_max != 0.).all():
		data_norm = (data - att_min) / att_max
	else:
		raise Exception("Zero!")

	if torch.isnan(data_norm).any():
		raise Exception("nans!")

	return data_norm, att_min, att_max


def normalize_masked_data(data, mask, att_min, att_max):
	# data = torch.clamp(data, min=att_min, max=att_max) * mask
	# scale = att_max
	scale = att_max - att_min
	scale = scale + (scale == 0) * 1e-8
	# we don't want to divide by zero
	if (scale != 0.).all(): 
		data_norm = (data - att_min) / scale
	else:
		raise Exception("Zero!")

	# set masked out elements back to zero 
	data_norm[mask == 0] = 0
	# data_norm = data_norm * mask

	# print(data_norm[torch.isnan(data_norm)])
	# print(data[torch.isnan(data_norm)])
	# print(mask[torch.isnan(data_norm)])

	if torch.isnan(data_norm).any():
		raise Exception("nans!")

	return data_norm

def normalize_masked_tp(data, att_min, att_max):
	scale = att_max - att_min
	scale = scale + (scale == 0) * 1e-8
	# we don't want to divide by zero
	if (scale != 0.).all():
		data_norm = (data - att_min) / scale
	else:
		raise Exception("Zero!")

	if torch.isnan(data_norm).any():
		raise Exception("nans!")

	return data_norm


def shift_outputs(outputs, first_datapoint = None):
	outputs = outputs[:,:,:-1,:]

	if first_datapoint is not None:
		n_traj, n_dims = first_datapoint.size()
		first_datapoint = first_datapoint.reshape(1, n_traj, 1, n_dims)
		outputs = torch.cat((first_datapoint, outputs), 2)
	return outputs

def split_and_patch_batch(data_dict, args, n_observed_tp, patch_indices, max_len=None):

	device = get_device(data_dict["data"])

	split_dict = {"tp_to_predict": data_dict["tp_to_predict"].clone(),
			"data_to_predict": data_dict["data_to_predict"].clone(),
			"mask_predicted_data": data_dict["mask_predicted_data"].clone()
			}
	
	observed_tp = data_dict["time_steps"].clone() # (n_observed_tp, )
	observed_data = data_dict["data"].clone() # (bs, n_observed_tp, D)
	observed_mask = data_dict["mask"].clone() # (bs, n_observed_tp, D)

	n_batch, n_tp, n_dim = observed_data.shape
	observed_tp_patches = observed_tp.view(1, 1, -1, 1).repeat(n_batch, args.npatch, 1, n_dim)
	observed_data_patches = observed_data.view(n_batch, 1, n_tp, n_dim).repeat(1, args.npatch, 1, 1)
	observed_mask_patches = observed_mask.view(n_batch, 1, n_tp, n_dim).repeat(1, args.npatch, 1, 1)

	
	max_patch_len = args.maxlen
	for i in range(args.npatch):
		indices = patch_indices[i]
		if(len(indices) == 0): continue
		st_ind, ed_ind = indices[0], indices[-1]
		n_data_points = observed_mask[:, st_ind:ed_ind+1].sum(dim=1).max().item()
		max_patch_len = max(max_patch_len, int(n_data_points))

	observed_mask_patches_fill = torch.zeros_like(observed_mask_patches, dtype=observed_mask.dtype) # n_batch, npacth, n_tp, n_dim
	patch_indices_fianl = torch.full((n_batch, args.npatch, max_patch_len, n_dim), n_tp).to(device) # n_batch, npacth, max_patch_len, n_dim
	observed_mask_patches_fill_reindex = torch.zeros_like(patch_indices_fianl, dtype=observed_mask.dtype)
	aux_tensor = torch.arange(max_patch_len).view(1, max_patch_len, 1).repeat(n_batch, 1, n_dim).to(device)
	for i in range(args.npatch):
		indices = patch_indices[i]
		if(len(indices) == 0): continue
		st_ind, ed_ind = indices[0], indices[-1]
		observed_mask_patches_fill[:, i, st_ind:ed_ind+1] = observed_mask[:, st_ind:ed_ind+1, :]
		L = observed_mask[:, st_ind:ed_ind+1, :].sum(dim=1, keepdim=True) # (bs, 1, D)
		# print(L[0,0])
		# print(aux_tensor.permute(0, 2, 1)[0])
		# print(((aux_tensor < L)[0].sum(dim=0) != L[0,0]).sum())
		observed_mask_patches_fill_reindex[:, i] = (aux_tensor < L)  # let first L[i] to be True
	
	### return a indices tuple like ([...], [...], [...], [...])
	mask_inds = torch.nonzero(observed_mask_patches_fill_reindex.permute(0,1,3,2), as_tuple=True) # reset indices
	ind_values = torch.nonzero(observed_mask_patches_fill.permute(0,1,3,2), as_tuple=True)[-1] # original indices of dimension 2
	# print(mask_inds)
	# print(torch.nonzero(observed_mask_patches_fill.permute(0,1,3,2), as_tuple=True))

	### fill n_tp if the number of observed points are less than max_patch_len
	# patch_indices_fianl.index_put_((mask_inds[:,0], mask_inds[:,1], mask_inds[:,2], mask_inds[:,3]), ind_values)
	patch_indices_fianl.index_put_((mask_inds[0], mask_inds[1], mask_inds[3], mask_inds[2]), ind_values)

	pad_zeros_data = torch.zeros([n_batch, args.npatch, 1, n_dim]).to(device)
	observed_tp_patches = torch.cat([observed_tp_patches, pad_zeros_data], dim=2).gather(2, patch_indices_fianl) # (n_batch, npatch, max_patch_len, n_dim)
	observed_data_patches = torch.cat([observed_data_patches, pad_zeros_data], dim=2).gather(2, patch_indices_fianl)
	observed_mask_patches = torch.cat([observed_mask_patches, pad_zeros_data], dim=2).gather(2, patch_indices_fianl)
	
	# print(observed_tp_patches[0][0][:2])
	# print(observed_data_patches[0][0][:2])
	# print(observed_mask_patches[0][0][:2])
	# print(observed_tp_patches[0][:][:][observed_mask_patches[0][:][:].bool()])
	# print(observed_data_patches[0][:][:][observed_mask_patches[0][:][:].bool()])

	split_dict["observed_tp"] = observed_tp_patches
	split_dict["observed_data"] = observed_data_patches
	split_dict["observed_mask"] = observed_mask_patches

	return split_dict


### version 2
def split_data_forecast(data_dict, dataset, n_observed_tp):

	device = get_device(data_dict["data"])

	# n_observed_tp = data_dict["data"].size(1) // 2
	# if dataset == "hopper":
	# 	n_observed_tp = data_dict["data"].size(1) // 3

	split_dict = {"observed_data": data_dict["data"][:,:n_observed_tp,:].clone(),
				"observed_tp": data_dict["time_steps"][:n_observed_tp].clone(),
				"data_to_predict": data_dict["data"][:,n_observed_tp:,:].clone(),
				"tp_to_predict": data_dict["time_steps"][n_observed_tp:].clone()}

	split_dict["observed_mask"] = None 
	split_dict["mask_predicted_data"] = None 
	split_dict["labels"] = None 

	if ("mask" in data_dict) and (data_dict["mask"] is not None):
		split_dict["observed_mask"] = data_dict["mask"][:, :n_observed_tp].clone()
		split_dict["mask_predicted_data"] = data_dict["mask"][:, n_observed_tp:].clone()

	# if ("labels" in data_dict) and (data_dict["labels"] is not None):
	# 	split_dict["labels"] = data_dict["labels"].clone()

	split_dict["mode"] = "forecast"

	return split_dict



def split_data_interp(data_dict):
	device = get_device(data_dict["data"])

	split_dict = {"observed_data": data_dict["data"].clone(),
				"observed_tp": data_dict["time_steps"].clone(),
				"data_to_predict": data_dict["data"].clone(),
				"tp_to_predict": data_dict["time_steps"].clone()}

	split_dict["observed_mask"] = None 
	split_dict["mask_predicted_data"] = None 
	split_dict["labels"] = None 

	if "mask" in data_dict and data_dict["mask"] is not None:
		split_dict["observed_mask"] = data_dict["mask"].clone()
		split_dict["mask_predicted_data"] = data_dict["mask"].clone()

	if ("labels" in data_dict) and (data_dict["labels"] is not None):
		split_dict["labels"] = data_dict["labels"].clone()

	split_dict["mode"] = "interp"
	return split_dict



def add_mask(data_dict):
	data = data_dict["observed_data"]
	mask = data_dict["observed_mask"]

	if mask is None:
		mask = torch.ones_like(data).to(get_device(data))

	data_dict["observed_mask"] = mask
	return data_dict


def subsample_observed_data(data_dict, n_tp_to_sample = None, n_points_to_cut = None):
	# n_tp_to_sample -- if not None, randomly subsample the time points. The resulting timeline has n_tp_to_sample points
	# n_points_to_cut -- if not None, cut out consecutive points on the timeline.  The resulting timeline has (N - n_points_to_cut) points

	if n_tp_to_sample is not None:
		# Randomly subsample time points
		data, time_steps, mask = subsample_timepoints(
			data_dict["observed_data"].clone(), 
			time_steps = data_dict["observed_tp"].clone(), 
			mask = (data_dict["observed_mask"].clone() if data_dict["observed_mask"] is not None else None),
			n_tp_to_sample = n_tp_to_sample)

	if n_points_to_cut is not None:
		# Remove consecutive time points
		data, time_steps, mask = cut_out_timepoints(
			data_dict["observed_data"].clone(), 
			time_steps = data_dict["observed_tp"].clone(), 
			mask = (data_dict["observed_mask"].clone() if data_dict["observed_mask"] is not None else None),
			n_points_to_cut = n_points_to_cut)

	new_data_dict = {}
	for key in data_dict.keys():
		new_data_dict[key] = data_dict[key]

	new_data_dict["observed_data"] = data.clone()
	new_data_dict["observed_tp"] = time_steps.clone()
	new_data_dict["observed_mask"] = mask.clone()

	if n_points_to_cut is not None:
		# Cut the section in the data to predict as well
		# Used only for the demo on the periodic function
		new_data_dict["data_to_predict"] = data.clone()
		new_data_dict["tp_to_predict"] = time_steps.clone()
		new_data_dict["mask_predicted_data"] = mask.clone()

	return new_data_dict


def split_and_subsample_batch(data_dict, args, n_observed_tp):

	processed_dict = split_data_forecast(data_dict, args.dataset, n_observed_tp)

	# add mask
	processed_dict = add_mask(processed_dict)

	# Subsample points or cut out the whole section of the timeline
	# if (args.sample_tp is not None) or (args.cut_tp is not None):
	# 	processed_dict = subsample_observed_data(processed_dict, 
	# 		n_tp_to_sample = args.sample_tp, 
	# 		n_points_to_cut = args.cut_tp)

	# if (args.sample_tp is not None):
	# 	processed_dict = subsample_observed_data(processed_dict, 
	# 		n_tp_to_sample = args.sample_tp)
	return processed_dict


def compute_loss_all_batches(model,
	test_dataloader, args,
	n_batches, experimentID, device,
	n_traj_samples = 1, kl_coef = 1., 
	max_samples_for_eval = None):

	total = {}
	total["loss"] = 0
	total["likelihood"] = 0
	total["mse"] = 0
	total["kl_first_p"] = 0
	total["std_first_p"] = 0
	total["pois_likelihood"] = 0
	total["ce_loss"] = 0

	n_test_samples = 0
	
	classif_predictions = torch.Tensor([]).to(device)
	all_test_labels =  torch.Tensor([]).to(device)

	for i in range(n_batches):
		# print("Computing loss... " + str(i))
		
		batch_dict = get_next_batch(test_dataloader)
		bs = batch_dict["observed_data"].shape[0]

		results  = model.compute_all_losses(batch_dict,
			n_traj_samples = n_traj_samples, kl_coef = kl_coef)

		if args.classif:
			n_labels = model.n_labels #batch_dict["labels"].size(-1)
			n_traj_samples = results["label_predictions"].size(0)

			classif_predictions = torch.cat((classif_predictions, 
				results["label_predictions"].reshape(n_traj_samples, -1, n_labels)),1)
			all_test_labels = torch.cat((all_test_labels, 
				batch_dict["labels"].reshape(-1, n_labels)),0)

		for key in total.keys(): 
			if key in results:
				var = results[key]
				if isinstance(var, torch.Tensor):
					var = var.detach()
				total[key] += var*bs

		n_test_samples += bs

		# for speed
		if max_samples_for_eval is not None:
			if n_batches * batch_size >= max_samples_for_eval:
				break

	if n_test_samples > 0:
		for key, value in total.items():
			total[key] = total[key] / n_test_samples
 
	if args.classif:
		if args.dataset == "physionet":
			#all_test_labels = all_test_labels.reshape(-1)
			# For each trajectory, we get n_traj_samples samples from z0 -- compute loss on all of them
			all_test_labels = all_test_labels.repeat(n_traj_samples,1,1)


			idx_not_nan = ~torch.isnan(all_test_labels)
			classif_predictions = classif_predictions[idx_not_nan]
			all_test_labels = all_test_labels[idx_not_nan]

			dirname = "plots/" + str(experimentID) + "/"
			os.makedirs(dirname, exist_ok=True)
			
			total["auc"] = 0.
			if torch.sum(all_test_labels) != 0.:
				print("Number of labeled examples: {}".format(len(all_test_labels.reshape(-1))))
				print("Number of examples with mortality 1: {}".format(torch.sum(all_test_labels == 1.)))

				# Cannot compute AUC with only 1 class
				total["auc"] = sk.metrics.roc_auc_score(all_test_labels.cpu().numpy().reshape(-1), 
					classif_predictions.cpu().numpy().reshape(-1))
			else:
				print("Warning: Couldn't compute AUC -- all examples are from the same class")
		
		if args.dataset == "activity":
			all_test_labels = all_test_labels.repeat(n_traj_samples,1,1)

			labeled_tp = torch.sum(all_test_labels, -1) > 0.

			all_test_labels = all_test_labels[labeled_tp]
			classif_predictions = classif_predictions[labeled_tp]

			# classif_predictions and all_test_labels are in on-hot-encoding -- convert to class ids
			_, pred_class_id = torch.max(classif_predictions, -1)
			_, class_labels = torch.max(all_test_labels, -1)

			pred_class_id = pred_class_id.reshape(-1) 

			total["accuracy"] = sk.metrics.accuracy_score(
					class_labels.cpu().numpy(), 
					pred_class_id.cpu().numpy())
	return total

def check_mask(data, mask):
	#check that "mask" argument indeed contains a mask for data
	n_zeros = torch.sum(mask == 0.).cpu().numpy()
	n_ones = torch.sum(mask == 1.).cpu().numpy()

	# mask should contain only zeros and ones
	assert((n_zeros + n_ones) == np.prod(list(mask.size())))

	# all masked out elements should be zeros
	assert(torch.sum(data[mask == 0.] != 0.) == 0)


def get_max_patch_len(total_dataset, args):
    """전체 데이터셋을 스캔하여 가장 긴 패치의 길이를 계산합니다."""
    global_max_patch_len = 0
    
    for _, tt, _, _ in total_dataset:
        # 각 데이터에서 관측 구간의 시간축(observed_tp)만 추출
        observed_tp = tt[torch.lt(tt, args.history)]
        
        # 패치 생성 로직을 그대로 사용하여 패치별 길이 계산
        st, ed = 0, args.patch_size
        for i in range(args.npatch):
            if i == args.npatch - 1:
                inds = torch.where((observed_tp >= st) & (observed_tp <= ed))[0]
            else:
                inds = torch.where((observed_tp >= st) & (observed_tp < ed))[0]
            
            # 현재 패치 길이와 기록된 최대 길이 중 더 큰 값을 저장
            global_max_patch_len = max(global_max_patch_len, len(inds))
            
            st += args.stride
            ed += args.stride
            
    print(f"Global Max Patch Length Found: {global_max_patch_len}")
    return global_max_patch_len