import hydra
import wandb
import math
import time
import numpy as np
import torch
import random
import matplotlib.pyplot as plt
from torch.nn import functional as F

from torch_seq_moo.algorithms.base import BaseAlgorithm
from torch_seq_moo.algorithms.mogfn_utils.utils import mean_pairwise_distances, generate_simplex, thermometer, plot_pareto, pareto_frontier
from torch_seq_moo.utils import str_to_tokens, tokens_to_str
from torch_seq_moo.metrics import get_all_metrics

from torch.distributions import Categorical
from tqdm import tqdm


class MOGFN(BaseAlgorithm):
    def __init__(self, cfg, tokenizer, task_cfg, **kwargs):
        super(MOGFN, self).__init__(cfg, tokenizer, task_cfg)
        self.setup_vars(kwargs)
        self.init_policy()

    def setup_vars(self, kwargs):
        cfg = self.cfg
        # Task stuff
        self.max_len = cfg.max_len
        self.min_len = cfg.min_len
        self.obj_dim = self.task_cfg.obj_dim
        # GFN stuff
        self.train_steps = cfg.train_steps
        self.random_action_prob = cfg.random_action_prob
        self.batch_size = cfg.batch_size
        self.reward_min = cfg.reward_min
        self.therm_n_bins = cfg.therm_n_bins
        self.beta_use_therm = cfg.beta_use_therm
        self.pref_use_therm = cfg.pref_use_therm
        self.gen_clip = cfg.gen_clip
        self.sampling_temp = cfg.sampling_temp
        self.sample_beta = cfg.sample_beta
        self.beta_cond = cfg.beta_cond
        self.pref_cond = cfg.pref_cond
        self.beta_scale = cfg.beta_scale
        self.beta_shape = cfg.beta_shape
        self.pref_alpha = cfg.pref_alpha
        self.beta_max = cfg.beta_max
        # Eval Stuff
        self._hv_ref = None
        self._ref_point = np.array([0] * self.obj_dim)
        self.eval_metrics = cfg.eval_metrics
        self.eval_freq = cfg.eval_freq
        self.k = cfg.k
        self.num_samples = cfg.num_samples
        self.eos_char = "[SEP]"
        self.pad_tok = self.tokenizer.convert_token_to_id("[PAD]")
        self.simplex = generate_simplex(self.obj_dim, cfg.simplex_bins)
        self.val_data = {
            'x': ["A", "AA", "AR", "RA", "RR", "R"]
        }

    def init_policy(self):
        cfg = self.cfg
        pref_dim = self.therm_n_bins * self.obj_dim if self.pref_use_therm else self.obj_dim
        beta_dim = self.therm_n_bins if self.beta_use_therm else 1
        cond_dim = pref_dim + beta_dim if self.beta_cond else pref_dim
        self.model = hydra.utils.instantiate(cfg.model, cond_dim=cond_dim)

        self.device = torch.device("cuda") if torch.cuda.is_available() else torch.device("cpu")
        self.model.to(self.device)
        self.opt = torch.optim.Adam(self.model.parameters(), cfg.pi_lr, weight_decay=cfg.wd,
                            betas=(0.9, 0.999))


    def optimize(self, task, init_data=None):
        """
        optimize the task involving multiple objectives (all to be maximized) with 
        optional data to start with
        """
        losses, rewards = [], []
        hv, r2, hsri, rs = 0., 0., 0., np.zeros(self.obj_dim)
        pb = tqdm(range(self.train_steps))
        desc_str = "Evaluation := Reward: {:.3f} HV: {:.3f} R2: {:.3f} HSRI: {:.3f} | Train := Loss: {:.3f} Rewards: {:.3f}"
        pb.set_description(desc_str.format(rs.mean(), hv, r2, hsri, sum(losses[-10:]) / 10, sum(rewards[-10:]) / 10))

        for i in pb:
            # if i==300:
            #     import pdb;pdb.set_trace();
            loss, r = self.train_step(task, self.batch_size)
            # with torch.no_grad():
            #     val_loss = self.val_step(task, self.batch_size)
            #     print(val_loss)
            losses.append(loss)
            rewards.append(r)
            
            if i != 0 and i % self.eval_freq == 0:
                with torch.no_grad():
                    # import pdb; pdb.set_trace();
                    samples, all_rews, rs, mo_metrics, topk_metrics, fig = self.evaluation(task, plot=True)
                hv, r2, hsri = mo_metrics["hypervolume"], mo_metrics["r2"], mo_metrics["hsri"]
                
                self.log(dict(
                    topk_rewards=topk_metrics[0].mean(),
                    topk_diversity=topk_metrics[1].mean(),
                    sample_r=rs.mean()
                ), commit=False)

                self.log({key: val for key, val in mo_metrics.items()}, commit=False)

                if fig is not None:
                    self.log(dict(
                        pareto_front=fig
                    ), commit=False)
                table = wandb.Table(columns = ["Sequence", "Rewards", "Prefs"])
                for sample, rew, pref in zip(samples, all_rews, self.simplex):
                    table.add_data(str(sample), str(rew), str(pref))
                self.log({"generated_seqs": table})
            self.log(dict(
                train_loss=loss,
                train_rewards=r,
            ))
            pb.set_description(desc_str.format(rs.mean(), hv, r2, hsri, sum(losses[-10:]) / 10, sum(rewards[-10:]) / 10))
        
        return {
            'losses': losses,
            'train_rs': rewards,
            'hypervol_rel': hv
        }
    
    def train_step(self, task, batch_size):
        cond_var, (prefs, beta) = self._get_condition_var(train=True, bs=batch_size)
        states, logprobs = self.sample(batch_size, cond_var)
        (s, a, d, r, tidx), rs = self.get_batch_states(states, prefs, task)
        self.opt.zero_grad()
        cond_var = torch.tile(cond_var[0], (s.shape[0], 1))
        flows = self.model(s.swapaxes(0,1), cond_var, mask=s.eq(0))
        qsa = torch.logaddexp(flows[tidx, a], torch.log(torch.tensor(self.reward_min).to(self.device)))
        qsp = torch.logsumexp(flows[tidx+1], 1)
        qsp = qsp * (1-d) - (1000) * d
        # import pdb; pdb.set_trace();
        outflow = torch.logaddexp(beta * torch.log(r.clamp(min=self.reward_min).to(self.device)), qsp)

        loss = (qsa - outflow).pow(2)
        # leaf_loss = (loss * d).sum() / d.sum()
        # flow_loss = (loss * (1-d)).sum() / (1-d).sum()
        # loss = 5 * leaf_loss + flow_loss
        loss = loss.mean()

        loss.backward()
        torch.nn.utils.clip_grad_norm_(self.model.parameters(), self.gen_clip)
        self.opt.step()
        return loss.item(), rs.mean()

    def sample(self, episodes, cond_var=None, train=True):
        states = [''] * episodes
        traj_logprob = torch.zeros(episodes).to(self.device)
        
        if cond_var is None:
            cond_var, _ = self._get_condition_var(train=train, bs=episodes)
        
        active_mask = torch.ones(episodes).bool().to(self.device)
        x = str_to_tokens(states, self.tokenizer, use_sep=False).to(self.device).t()[:1]
        all_states = x.clone().detach().unsqueeze(0)
        all_actions = None
        all_dones = torch.logical_not(active_mask).to(self.device)
        lens = torch.zeros(episodes).long().to(self.device)
        uniform_pol = torch.empty(episodes).fill_(self.random_action_prob).to(self.device)

        for t in (range(self.max_len) if episodes > 0 else []):
            logits = self.model(x, cond_var, mask=x.lt(0).t())
            if t <= self.min_len:
                logits[:, 0] = -1000 # Prevent model from stopping
                                     # without having output anything

            cat = Categorical(logits=logits / self.sampling_temp)
            actions = cat.sample()
            if train and self.random_action_prob > 0:
                uniform_mix = torch.bernoulli(uniform_pol).bool()
                actions = torch.where(uniform_mix, torch.randint(int(t <= self.min_len), logits.shape[1], (episodes, )).to(self.device), actions)
            # if t==0:
            #     print(cat.probs)
            log_prob = cat.log_prob(actions) * active_mask
            traj_logprob += log_prob

            actions_apply = torch.where(torch.logical_not(active_mask), torch.zeros(episodes).to(self.device).long(), actions + 4)
            active_mask = torch.where(active_mask, actions != 0, active_mask)

            x = torch.cat((x, actions_apply.unsqueeze(0)), axis=0)
            if active_mask.sum() == 0:
                break
        states = tokens_to_str(x.t(), self.tokenizer)
        return states, traj_logprob

    def get_batch_states(self, states, prefs, task):
        with torch.no_grad():
            rs = self.process_reward(states, prefs, task)
        episodes_per_step = len(states)
        traj_states = [[] for i in range(episodes_per_step)]
        traj_actions = [[] for i in range(episodes_per_step)]
        traj_rewards = [[] for i in range(episodes_per_step)]
        traj_dones = [[] for i in range(episodes_per_step)]
        for i in range(episodes_per_step):
            traj_states[i].append('')
            for c, a in zip(states[i] + "%", str_to_tokens([states[i]+ "%" ], self.tokenizer, use_sep=False)[0][1:]-4):
                if len(traj_states[i][-1]) < self.max_len:
                    traj_states[i].append(traj_states[i][-1] + c)
                    traj_actions[i].append(a)
                    traj_rewards[i].append(0 if c != "%" else rs[i])
                    traj_dones[i].append(float(c == "%"))
            if traj_states[i][-1][-1] != "%":
                traj_rewards[i][-1] = rs[i]
                traj_dones[i][-1] = 1
        tidx = [[-2]]
        # The index of s in the concatenated trajectories
        for i in traj_states:
            tidx.append(torch.arange(len(i) - 1) + tidx[-1][-1] + 2)
        # import pdb; pdb.set_trace();
        tidx = torch.cat(tidx[1:]).to(self.device)
        s = str_to_tokens(sum(traj_states, []), self.tokenizer, use_sep=False).to(self.device)
        a = torch.tensor(sum(traj_actions, [])).to(self.device)
        d = torch.tensor(sum(traj_dones, [])).to(self.device)
        r = torch.tensor(sum(traj_rewards, [])).to(self.device)
        return (s, a, d, r, tidx), rs
    
    def process_reward(self, seqs, prefs, task, rewards=None):
        if rewards is None:
            rewards = task.score(seqs)
        reward = (torch.tensor(prefs) * (rewards)).sum(axis=1)
        return reward

    def evaluation(self, task, plot=False):
        new_candidates = []
        r_scores = [] 
        all_rewards = []
        topk_rs = []
        topk_div = []
        for prefs in self.simplex:
            cond_var, (_, beta) = self._get_condition_var(prefs=prefs, train=False, bs=self.num_samples)
            samples, _ = self.sample(self.num_samples, cond_var, train=False)
            rewards = task.score(samples)
            r = self.process_reward(samples, prefs, task, rewards=rewards)
            
            # topk metrics
            topk_r, topk_idx = torch.topk(r, self.k)
            samples = np.array(samples)
            topk_seq = samples[topk_idx].tolist()
            edit_dist = mean_pairwise_distances(topk_seq)
            topk_rs.append(topk_r.mean().item())
            topk_div.append(edit_dist)
            
            # top 1 metrics
            max_idx = r.argmax()
            new_candidates.append(samples[max_idx])
            all_rewards.append(rewards[max_idx])
            r_scores.append(r.max().item())

        r_scores = np.array(r_scores)
        all_rewards = np.array(all_rewards)
        new_candidates = np.array(new_candidates)
        
        # filter to get current pareto front 
        pareto_candidates, pareto_targets = pareto_frontier(new_candidates, all_rewards, maximize=True)
        
        mo_metrics = get_all_metrics(pareto_targets, self.eval_metrics, hv_ref=self._ref_point, r2_prefs=self.simplex, num_obj=self.obj_dim)
        fig = plot_pareto(pareto_targets, all_rewards, pareto_only=False) if plot else None        
        return new_candidates, all_rewards, r_scores, mo_metrics, (np.array(topk_rs), np.array(topk_div)), fig

    def val_step(self, task, batch_size):
        overall_loss = 0.
        for pref in self.simplex:
            cond_var, (prefs, beta) = self._get_condition_var(prefs=pref, train=False, bs=batch_size)
            num_batches = len(self.val_data['x']) // self.batch_size
            losses = 0
            for i in range(num_batches):
                states = self.val_data['x'][i * self.batch_size:(i+1) * self.batch_size]
                
                (s, a, d, r, tidx), rs = self.get_batch_states(states, prefs, task)
                cond_var = torch.tile(cond_var[0], (s.shape[0], 1))
                flows = self.model(s.swapaxes(0,1), cond_var, mask=s.eq(0))
                qsa = torch.logaddexp(flows[tidx, a], torch.log(torch.tensor(self.reward_min).to(self.device)))
                qsp = torch.logsumexp(flows[tidx+1], 1)
                qsp = qsp * (1-d) - (1000) * d
                # import pdb; pdb.set_trace();
                outflow = torch.logaddexp(beta * torch.log(r.clamp(min=self.reward_min).to(self.device)), qsp)

                loss = (qsa - outflow).pow(2)
                loss = loss.mean()
                losses += loss.item()
            overall_loss += (losses / num_batches)
        return overall_loss / len(self.simplex)

    def _get_condition_var(self, prefs=None, beta=None, train=True, bs=None):
        if prefs is None:
            if not train:
                prefs = self.simplex[0]
            else:
                prefs = np.random.dirichlet([self.pref_alpha]*self.obj_dim)
        if beta is None:
            if train:
                beta = float(np.random.randint(1, self.beta_max+1)) if self.beta_cond else self.sample_beta
            else:
                beta = self.sample_beta
        
        # encoding
        if self.pref_use_therm:
            prefs_enc = thermometer(torch.from_numpy(prefs), self.therm_n_bins, 0, 1) 
        else: 
            prefs_enc = torch.from_numpy(prefs)
        
        if self.beta_use_therm:
            beta_enc = thermometer(torch.from_numpy(np.array([beta])), self.therm_n_bins, 0, self.beta_max) 
        else:
            beta_enc = torch.from_numpy(np.array([beta]))
        
        # construct final conditioning variable
        if self.beta_cond:
            cond_var = torch.cat((prefs_enc.view(-1), beta_enc.view(-1))).float().to(self.device)
        else:
            cond_var = prefs_enc.view(-1).float().to(self.device)
        if bs:
            cond_var = torch.tile(cond_var.unsqueeze(0), (bs, 1))
        return cond_var, (prefs, beta)