import time
import os
import torch
import numpy as np
from metrics import averageMeter
from loss import get_loss
import pandas as pd
from utils import Info_Geo_Riem_metric_diagonal_forloop

class TeacherTrainer:
    """Trainer for a conventional iterative training of model"""
    def __init__(self, training_cfg, device):
        self.cfg = training_cfg
        self.device = device
        self.d_val_result = {}
        self.loss = get_loss(training_cfg['loss'], device=device)

    def train(self, model, opt, d_dataloaders, student_loaders=None, logger=None, logdir='', *args, **kwargs):
        cfg = self.cfg
        ### newly added ###
        test_recon = cfg.get('test_recon', False)
        print("test_recon: ", test_recon)
        student_eval_period = cfg.get('student_eval_period', 1)
        print("student_eval_period: ", student_eval_period)
        ################
        best_val_loss = np.inf
        best_classify_acc = 0
        time_meter = averageMeter()
        i = 0
        train_loader = d_dataloaders['training']
        
        ### newly added ###
        # precalculate the Riemannian metric if random_jitter is false
        use_pre_metric = False
        train_dataset = train_loader.dataset
        if not train_dataset.random_jitter and ((model.contractive_reg and not model.contractive_use_identity) or (model.denoise_reg and not model.denoise_use_identity)):
            use_pre_metric = True
            data = torch.tensor(train_dataset.data).type(torch.float32).permute(0,2,1) # (batch, num_pts, 3) -> (batch, 3, num_pts)
            data = data.to(self.device)
            train_dataset.blkdiag_metric = Info_Geo_Riem_metric_diagonal_forloop(data, 16, sigma=model.sigma, block=True).cpu().numpy()
        
        print("iters per epoch: ", len(train_loader))
        if model.contractive_reg:
            if model.use_identity:
                if model.contractive_use_identity:
                    print('RCAE')
            else:
                if model.contractive_use_identity:
                    print('GCAE')
                else:
                    print('GRCAE')
        elif model.denoise_reg:
            if model.denoise_use_identity:
                print('DAE')
            else:
                print('GDAE')
        else:
            print('AE')
        if model.fm_reg:
            print('+ FM reg.')
        ################
        
        for i_epoch in range(cfg.n_epoch):
            for output in train_loader:
                i += 1
                model.train()
                x = output[0].to(self.device)
                
                ### newly added ###
                diag_metric = output[2]
                if use_pre_metric:
                    diag_metric = diag_metric.to(self.device)
                else:
                    diag_metric = None
                ################
                
                start_ts = time.time()
                #d_train = model.train_step(x, optimizer=opt, loss=self.loss)
                ### newly added ###
                d_train = model.train_step(x, optimizer=opt, loss=self.loss, diag_metric = diag_metric)
                
                #if i % cfg.print_interval == 0:
                #    print(f"Avg Loss: {d_train['loss']:.4f} Reg Loss: {d_train['loss_contractive']:.4f}")
                ################

                time_meter.update(time.time() - start_ts)
                logger.process_iter_train(d_train)

                if i % cfg.print_interval == 0:
                    d_train = logger.summary_train(i)
                    print(f"Iter [{i:d}] Avg Loss: {d_train['loss/train_loss_']:.4f} Elapsed time: {time_meter.sum:.4f}")
                    time_meter.reset()
    
            ## teaching
            if (i_epoch + 1) % student_eval_period == 0:
                df = pd.DataFrame()
                for key, item in student_loaders.items():
                    ### newly modified ###
                    if not test_recon:
                        student_train_loader = item['training']
                    else:
                        student_train_loader = []
                    ###################
                    student_test_loader = item['test']
                    student_dict = model.student_evaluation_step(
                        student_train_loader, student_test_loader, self.device)
                    student_dict['dataset'] = key

                    df = df.append(
                        student_dict, ignore_index=True
                    )

                    savedir = os.path.join(logdir, 'teaching_results')
                    if not os.path.exists(savedir):
                        os.makedirs(savedir)

                    df.to_csv(os.path.join(savedir, f'df_teaching_results_epoch_{i_epoch}.csv'))

            if (i_epoch+1) % 100 == 0:
                self.save_model(model, logdir, best=False, i_epoch=i_epoch+1)

        self.save_model(model, logdir, best=False, last=True)
        return model, best_val_loss

    def save_model(self, model, logdir, best=False, last=False, i_iter=None, i_epoch=None, best_classify=False):
        if last:
            pkl_name = "model_last.pkl"
        else:
            if best:
                pkl_name = "model_best.pkl"
            elif best_classify:
                pkl_name = "model_best_classify.pkl"
            else:
                if i_iter is not None:
                    pkl_name = "model_iter_{}.pkl".format(i_iter)
                else:
                    pkl_name = "model_epoch_{}.pkl".format(i_epoch)
        state = {"epoch": i_epoch, "model_state": model.state_dict(), 'iter': i_iter}
        save_path = os.path.join(logdir, pkl_name)
        torch.save(state, save_path)
        print(f'Model saved: {pkl_name}')
        