import numpy as np
import torch
import wandb
import comet_ml
from tqdm import tqdm
import math
import torch.nn.functional as F

from recurrent_baselines.Mamba.mingpt.model_vizdoom import GPT, GPTConfig
from recurrent_baselines.Mamba.vizdoom import val_vizdoom

from VizDoom.VizDoom_src.utils import z_normalize, inverse_z_normalize
from TMaze_new.TMaze_new_src.utils import seeds_list

torch.backends.cudnn.benchmark = True

from scipy.stats import sem

import warnings
warnings.filterwarnings('ignore')
import torch.nn as nn

reds = [2, 3, 6, 8, 9, 10, 11, 14, 15, 16, 17, 18, 20, 21, 25, 26, 27, 28, 29, 31, 38, 40, 41, 42, 45,
        46, 49, 50, 51, 52, 53, 54, 55, 58, 59, 60, 61, 63, 64, 67, 68, 70, 72, 73, 74, 77, 80, 82, 84, 
        86, 88, 89, 90, 91, 92, 97, 98, 99, 100, 101, 103, 106, 108, 109, 113, 115, 116, 117, 120, 
        123, 124, 125, 126, 127, 128, 129, 133, 134, 136, 139, 140, 142, 144, 145, 147, 148, 151, 152, 
        153, 154, 156, 157, 158, 159, 161, 164, 165, 170, 171, 173]

greens = [0, 1, 4, 5, 7, 12, 13, 19, 22, 23, 24, 30, 32, 33, 34, 35, 36, 37, 39, 43, 44, 47, 48, 56, 57,
          62, 65, 66, 69, 71, 75, 76, 78, 79, 81, 83, 85, 87, 93, 94, 95, 96, 102, 104, 105, 107, 110, 111, 
          112, 114, 118, 119, 121, 122, 130, 131, 132, 135, 137, 138, 141, 143, 146, 149, 150, 155, 160, 162, 
          163, 166, 167, 168, 169, 172, 175, 176, 177, 182, 183, 187, 190, 192, 193, 195, 199, 204, 206, 208, 
          209, 210, 212, 215, 216, 218, 219, 220, 221, 223, 224, 225]


def train(ckpt_path, config, train_dataloader, mean, std, max_segments, experiment):
    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')
    # device = 'cpu'

    episode_timeout = config["online_inference_config"]["episode_timeout"]

    MEAN = mean
    STD = std

    mconf = GPTConfig(
        vocab_size=None,
        block_size=config['training_config']['context_length'],
        n_layer=config['model_config']['n_layer'],
        # n_head=config['model_config']['n_head'],
        model_type=config['model_config']['model_type'],
        n_embd=config['model_config']['d_model'],
        token_mixer=config['model_config']['token_mixer'],
        # window_size=config['model_config']['conv_window_size'],
        # conv_proj=config['model_config']['conv_proj'],
        max_timestep=config['training_config']['context_length']-1,
        )
    model = GPT(mconf)

    print(config)

    wandb_step  = 0

    optimizer = torch.optim.AdamW(model.parameters(), 
                                  lr=config["training_config"]["learning_rate"], weight_decay=config["training_config"]["weight_decay"], 
                                  betas=(config["training_config"]["beta_1"], config["training_config"]["beta_2"]))

    raw_model = model.module if hasattr(model, "module") else model
        
    model.to(device)
    model.train()
    
    
    wwandb = config["wandb_config"]["wwandb"]
    wcomet = config['wandb_config']['wcomet']

    print(f"model parameters: {sum(p.numel() for p in list(model.parameters()))}")
    it_counter = 0
    
    pbar = tqdm(range(config["training_config"]["epochs"]))


    tokens = 0

    for epoch in pbar:
        train_imgs = []
        is_train = True
        model.train()
        for it, batch in enumerate(train_dataloader):
            s, a, rtg, d, timesteps, masks = batch

            if config["data_config"]["normalize"] == 2:
                _b, _l, _c, _h, _w = s.shape
                s = s.reshape(_b*_l, _c, _h, _w)
                s = z_normalize(s, MEAN, STD)
                s = s.reshape(_b, _l, _c, _h, _w)
            
            x1 = s[:, :, :].to(device)
            y1 = a[:, :, :].to(device).float()
            r1 = rtg[:,:,:][:, :, :].to(device).float() 
            t1 = timesteps[:, :].to(device)
            masks1 = masks[:, :].to(device)
            

            with torch.set_grad_enabled(is_train):
                
                logits, train_loss = model(states=x1, actions=y1, targets=y1, rtgs=r1, timesteps=timesteps.unsqueeze(-1).to(device))
                
                if wwandb:
                    wandb.log({"train_loss":  train_loss.item()})
                elif wcomet:
                    experiment.log_metric("train_loss", train_loss.item(), step=it_counter)

            if is_train:
                model.zero_grad()
                optimizer.zero_grad()
                train_loss.backward(retain_graph=True)
                if config["training_config"]["grad_norm_clip"] is not None:
                    torch.nn.utils.clip_grad_norm_(model.parameters(), config["training_config"]["grad_norm_clip"])
                optimizer.step()

                # * decay the learning rate based on our progress
                tokens += (y1 >= 0).sum()
                if tokens < config["training_config"]["warmup_steps"]:
                    # linear warmup
                    lr_mult = float(tokens) / float(max(1, config["training_config"]["warmup_steps"]))
                else:
                    # cosine learning rate decay
                    progress = float(tokens - config["training_config"]["warmup_steps"]) / float(max(1, config["training_config"]["final_tokens"] - config["training_config"]["warmup_steps"]))
                    lr_mult = max(config["training_config"]["lr_end_factor"], 0.5 * (1.0 + math.cos(math.pi * progress)))
                lr = config["training_config"]["learning_rate"] * lr_mult
                for param_group in optimizer.param_groups:
                    param_group['lr'] = lr



                lr = optimizer.state_dict()['param_groups'][0]['lr']
                if wwandb:
                    wandb.log({"learning_rate": lr})
                elif wcomet:
                    experiment.log_metric("learning_rate", lr, step=it_counter)

            it_counter += 1

            pbar.set_description(f"ep {epoch+1} it {it} tTotal {train_loss.item():.2f} lr {lr:e} tokens, M {(tokens/1e6):.2f}")

        if wwandb:
            wandb.log({"epochs": epoch+1})
        elif wcomet:
            experiment.log_metrics({"epochs": epoch+1}, step=it_counter)
        
        # ? Save
        if epoch == config["training_config"]["epochs"] - 1:
            if config["training_config"]["online_inference"] == True:
                if config["model_config"]["mode"] == 'doom':
                    model.eval()
                    with torch.no_grad():
                        FRAME_SKIP = 2
                        def optimize_pillar(color, seeds, config, wwandb, wcomet):
                            for ret in [config["online_inference_config"]["desired_return_1"]]:
                                returns = []
                                ts = []
                                attn_map_received = False
                                for i in range(len(seeds)):
                                    episode_return, act_list, t, _, _, attn_map = val_vizdoom.get_returns_VizDoom(model=model, ret=ret, seed=seeds[i], 
                                                                                            episode_timeout=episode_timeout, 
                                                                                            context_length=config["training_config"]["context_length"], 
                                                                                            device=device, 
                                                                                            act_dim=5, 
                                                                                            config=config,
                                                                                            mean=MEAN,
                                                                                            std=STD,
                                                                                            use_argmax=config["online_inference_config"]["use_argmax"],
                                                                                            create_video=False)

                                    returns.append(episode_return)
                                    t *= FRAME_SKIP
                                    ts.append(t)

                                    pbar.set_description(f"Online inference {color} {ret}: [{i+1} / {len(seeds)}] Time: {t}, Return: {episode_return:.2f}")

                                returns_mean = np.mean(returns)
                                lifetime_mean = np.mean(ts)

                                if wwandb:
                                    wandb.log({f"LifeTimeMean_{color}_{ret}": lifetime_mean, 
                                            f"ReturnsMean_{color}_{ret}": returns_mean})
                                elif wcomet:
                                    experiment.log_metrics({f"LifeTimeMean_{color}_{ret}": lifetime_mean, 
                                                           f"ReturnsMean_{color}_{ret}": returns_mean}, step=it_counter)
    
                            return returns, ts

                        total_returns, total_ts = [], []
                        SKIP_RETURN = 4

                        # RED PILLAR
                        seeds_red = reds[::SKIP_RETURN]
                        red_returns, red_ts = optimize_pillar("red", seeds_red, config, wwandb, wcomet)
                        total_returns += red_returns
                        total_ts += red_ts

                        # GREEN PILLAR
                        seeds_green = greens[::SKIP_RETURN]
                        green_returns, green_ts = optimize_pillar("green", seeds_green, config, wwandb, wcomet)
                        total_returns += green_returns
                        total_ts += green_ts

                        total_returns = np.mean(total_returns)
                        total_ts = np.mean(total_ts)

                        if wwandb:
                            wandb.log({f"LifeTimeMean_{config['online_inference_config']['desired_return_1']}": total_ts, 
                                       f"ReturnsMean_{config['online_inference_config']['desired_return_1']}": total_returns})
                        elif wcomet:
                            experiment.log_metrics({f"LifeTimeMean_{config['online_inference_config']['desired_return_1']}": total_ts, 
                                                    f"ReturnsMean_{config['online_inference_config']['desired_return_1']}": total_returns}, step=it_counter)
                                
            model.train()
            wandb_step += 1 
            if wwandb:
                wandb.log({"checkpoint_step": epoch+1})
            elif wcomet:
                experiment.log_metrics({"checkpoint_step": epoch+1}, step=it_counter)
            torch.save(model.state_dict(), ckpt_path + str(epoch+1) + '_KTD.pth')
            
    return model