import os
import numpy as np
import torch
from torch.optim import Adam
from torch.utils.tensorboard import SummaryWriter
from rltorch.memory import MultiStepMemory, PrioritizedMemory

from model import TwinnedQNetwork, GaussianPolicy, RandomizedEnsembleNetwork
from utils import grad_false, hard_update, soft_update, to_batch, update_params, RunningMeanStats

from collections import deque
import itertools
import math

# 20210813
import torch.autograd.profiler as profiler
import sys

# TODO merge with agent.py
class SacAgent4Profile: # created in 20210813

    def __init__(self, env, log_dir, num_steps=3000000, batch_size=256,
                 lr=0.0003, hidden_units=[256, 256], memory_size=1e6,
                 gamma=0.99, tau=0.005, entropy_tuning=True, ent_coef=0.2,
                 multi_step=1, per=False, alpha=0.6, beta=0.4,
                 beta_annealing=0.0001, grad_clip=None, updates_per_step=1,
                 start_steps=10000, log_interval=10, target_update_interval=1,
                 eval_interval=1000, cuda=0, seed=0,
                 # added by TH 20210707
                 eval_runs=1, huber=0, layer_norm=0,
                 method=None, target_entropy=None, target_drop_rate=0.0, critic_update_delay=1):
        self.env = env

        torch.manual_seed(seed)
        np.random.seed(seed)
        self.env.seed(seed)
        torch.backends.cudnn.deterministic = True  # It harms a performance.
        torch.backends.cudnn.benchmark = False

        self.method = method
        self.critic_update_delay = critic_update_delay
        self.target_drop_rate = target_drop_rate

        self.device = torch.device("cuda:" + str(cuda) if torch.cuda.is_available() else "cpu")

        # policy
        self.policy = GaussianPolicy(
            self.env.observation_space.shape[0],
            self.env.action_space.shape[0],
            hidden_units=hidden_units).to(self.device)

        # Q functions
        kwargs_q = {"num_inputs": self.env.observation_space.shape[0],
                    "num_actions": self.env.action_space.shape[0],
                    "hidden_units": hidden_units,
                    "layer_norm": layer_norm,
                    "drop_rate": self.target_drop_rate}
        if self.method == "redq":
            self.critic = RandomizedEnsembleNetwork(**kwargs_q).to(self.device)
            self.critic_target = RandomizedEnsembleNetwork(**kwargs_q).to(self.device)
        else:
            self.critic = TwinnedQNetwork(**kwargs_q).to(self.device)
            self.critic_target = TwinnedQNetwork(**kwargs_q).to(self.device)
        if self.target_drop_rate <= 0.0:
            self.critic_target = self.critic_target.eval()
        # copy parameters of the learning network to the target network
        hard_update(self.critic_target, self.critic)
        # disable gradient calculations of the target network
        grad_false(self.critic_target)

        # optimizer
        self.policy_optim = Adam(self.policy.parameters(), lr=lr)
        if self.method == "redq":
            for i in range(self.critic.N):
                setattr(self, "q"+str(i)+"_optim",
                        Adam(getattr(self.critic, "Q"+str(i)).parameters(), lr=lr))
        else:
            self.q1_optim = Adam(self.critic.Q1.parameters(), lr=lr)
            self.q2_optim = Adam(self.critic.Q2.parameters(), lr=lr)

        if entropy_tuning:
            if not (target_entropy is None):
                self.target_entropy = torch.prod(torch.Tensor([target_entropy]).to(self.device)).item()
            else:
                # Target entropy is -|A|.
                self.target_entropy = -torch.prod(torch.Tensor(self.env.action_space.shape).to(self.device)).item()
            # We optimize log(alpha), instead of alpha.
            self.log_alpha = torch.zeros(1, requires_grad=True, device=self.device)
            self.alpha = self.log_alpha.exp()
            self.alpha_optim = Adam([self.log_alpha], lr=lr)
        else:
            # fixed alpha
            self.alpha = torch.tensor(ent_coef).to(self.device)

        if per:
            # replay memory with prioritied experience replay
            # See https://github.com/ku2482/rltorch/blob/master/rltorch/memory
            self.memory = PrioritizedMemory(
                memory_size, self.env.observation_space.shape,
                self.env.action_space.shape, self.device, gamma, multi_step,
                alpha=alpha, beta=beta, beta_annealing=beta_annealing)
        else:
            # replay memory without prioritied experience replay
            # See https://github.com/ku2482/rltorch/blob/master/rltorch/memory
            self.memory = MultiStepMemory(
                memory_size, self.env.observation_space.shape,
                self.env.action_space.shape, self.device, gamma, multi_step)

        self.log_dir = log_dir
        self.model_dir = os.path.join(log_dir, 'model')
        self.summary_dir = os.path.join(log_dir, 'summary')
        if not os.path.exists(self.model_dir):
            os.makedirs(self.model_dir)
        if not os.path.exists(self.summary_dir):
            os.makedirs(self.summary_dir)

        self.writer = SummaryWriter(log_dir=self.summary_dir)
        self.train_rewards = RunningMeanStats(log_interval)

        self.steps = 0
        self.learning_steps = 0
        self.episodes = 0
        self.num_steps = num_steps
        self.tau = tau
        self.per = per
        self.batch_size = batch_size
        self.start_steps = start_steps
        self.gamma_n = gamma ** multi_step
        self.entropy_tuning = entropy_tuning
        self.grad_clip = grad_clip
        self.updates_per_step = updates_per_step
        self.log_interval = log_interval
        self.target_update_interval = target_update_interval
        self.eval_interval = eval_interval
        #
        self.eval_runs = eval_runs
        self.huber = huber
        self.multi_step = multi_step

    def run(self):
        while True:
            self.train_episode()
            if self.steps > self.num_steps:
                break

    def is_update(self):
        return len(self.memory) > self.batch_size and self.steps >= self.start_steps

    def act(self, state):
        if self.start_steps > self.steps:
            action = self.env.action_space.sample()
        else:
            action = self.explore(state)
        return action

    def explore(self, state):
        # act with randomness
        state = torch.FloatTensor(state).unsqueeze(0).to(self.device)
        with torch.no_grad():
            action, _, _ = self.policy.sample(state)
        return action.cpu().numpy().reshape(-1)

    def exploit(self, state):
        # act without randomness
        state = torch.FloatTensor(state).unsqueeze(0).to(self.device)
        with torch.no_grad():
            _, _, action = self.policy.sample(state)
        return action.cpu().numpy().reshape(-1)

    def calc_current_q(self, states, actions, rewards, next_states, dones):
        curr_q1, curr_q2 = self.critic(states, actions)
        return curr_q1, curr_q2

    def calc_target_q(self, states, actions, rewards, next_states, dones):
        with torch.no_grad():
            next_actions, next_entropies, _ = self.policy.sample(next_states)
            next_q1, next_q2 = self.critic_target(next_states, next_actions)
            if self.method == "sac" or self.method == "redq":
                next_q = torch.min(next_q1, next_q2) + self.alpha * next_entropies
            elif self.method == "duvn":
                next_q = next_q1 + self.alpha * next_entropies # discard q2
            else:
                raise NotImplementedError()
        # rescale rewards by num step TH20210705
        target_q = (rewards / (self.multi_step * 1.0)) + (1.0 - dones) * self.gamma_n * next_q
        return target_q

    def train_episode(self):
        self.episodes += 1
        episode_reward = 0.
        episode_steps = 0
        done = False
        state = self.env.reset()

        while not done:
            with profiler.profile(record_shapes=False, use_cuda=torch.cuda.is_available(), profile_memory=True, with_stack=True) as prof:
                with profiler.record_function("Whole_lines3-11"):
                    action = self.act(state)
                    next_state, reward, done, _ = self.env.step(action)
                    self.steps += 1
                    episode_steps += 1
                    episode_reward += reward

                    # ignore done if the agent reach time horizons
                    # (set done=True only when the agent fails)
                    if episode_steps >= self.env._max_episode_steps:
                        masked_done = False
                    else:
                        masked_done = done

                    if self.per:
                        batch = to_batch(
                            state, action, reward, next_state, masked_done,
                            self.device)
                        with torch.no_grad():
                            curr_q1, curr_q2 = self.calc_current_q(*batch)
                        target_q = self.calc_target_q(*batch)
                        # fixed by tH20210715
                        error = (0.5 * torch.abs(curr_q1 - target_q) + 0.5 * torch.abs(curr_q2 - target_q)).item()
                        # We need to give true done signal with addition to masked done
                        # signal to calculate multi-step rewards.
                        self.memory.append(state, action, reward, next_state, masked_done, error, episode_done=done)
                    else:
                        # We need to give true done signal with addition to masked done
                        # signal to calculate multi-step rewards.
                        self.memory.append(state, action, reward, next_state, masked_done, episode_done=done)

                    if self.is_update():
                        self.learn()
            if self.is_update():
                res = prof.key_averages(group_by_stack_n=1).table(sort_by="cpu_time_total", row_limit=10)
                print(res)
                a = prof.key_averages(group_by_stack_n=1)
                #
                with open(self.log_dir + "/" + "cpu_profile.csv", "a") as f:
                    f.write(res)
                    f.flush()
                res = prof.key_averages(group_by_stack_n=1).table(sort_by="cpu_memory_usage", row_limit=10)
                print(res)
                #
                with open(self.log_dir + "/" + "memory_profile.csv", "a") as f:
                    f.write(res)
                    f.flush()
                if torch.cuda.is_available():
                    res = prof.key_averages(group_by_stack_n=1).table(sort_by="cuda_time_total", row_limit=10)
                    print(res)
                    #
                    with open(self.log_dir + "/" + "cuda_profile.csv", "a") as f:
                        f.write(res)
                        f.flush()
                    res = prof.key_averages(group_by_stack_n=1).table(sort_by="cuda_memory_usage", row_limit=10)
                    print(res)
                    #
                    with open(self.log_dir + "/" + "cuda_memory_profile.csv", "a") as f:
                        f.write(res)
                        f.flush()
                if not hasattr(self, "num_rec"):
                    self.num_rec = 0
                self.num_rec += 1
                if self.num_rec > 4:
                    num_param = sum(p.numel() for p in self.critic.parameters() if p.requires_grad)
                    num_param += sum(p.numel() for p in self.critic_target.parameters() if p.requires_grad)
                    #
                    with open(self.log_dir + "/" + "total_critc_params.csv", "a") as f:
                        f.write(str(num_param))
                        f.flush()


                    sys.exit()



            if self.steps % self.eval_interval == 0:
                self.evaluate()
                self.save_models()

            state = next_state

        # We log running mean of training rewards.
        self.train_rewards.append(episode_reward)

        if self.episodes % self.log_interval == 0:
            self.writer.add_scalar('reward/train', self.train_rewards.get(), self.steps)

        print(f'episode: {self.episodes:<4}  '
              f'episode steps: {episode_steps:<4}  '
              f'reward: {episode_reward:<5.1f}')

    def learn(self):
        self.learning_steps += 1

        # critic update
        with profiler.record_function("TrainingQs_lines4-8"):
            if (self.learning_steps - 1) % self.critic_update_delay == 0:
                for _ in range(self.updates_per_step):
                    if self.per:
                        # batch with indices and priority weights
                        batch, indices, weights = self.memory.sample(self.batch_size)
                    else:
                        batch = self.memory.sample(self.batch_size)
                        # set priority weights to 1 when we don't use PER.
                        weights = 1.

                    if self.method == "redq":
                        losses, errors, mean_q1, mean_q2 = self.calc_critic_4redq_loss(batch, weights)
                        for i in range(self.critic.N):
                            update_params(getattr(self, "q" + str(i) + "_optim"),
                                                  getattr(self.critic, "Q" + str(i)),
                                                  losses[i], self.grad_clip)
                    else:
                        q1_loss, q2_loss, errors, mean_q1, mean_q2 = self.calc_critic_loss(batch, weights)

                        update_params(self.q1_optim, self.critic.Q1, q1_loss, self.grad_clip)
                        update_params(self.q2_optim, self.critic.Q2, q2_loss, self.grad_clip)

                    if self.learning_steps % self.target_update_interval == 0:
                        soft_update(self.critic_target, self.critic, self.tau)

                    if self.per:
                        # update priority weights
                        self.memory.update_priority(indices, errors.cpu().numpy())

        # policy and alpha update
        with profiler.record_function("TrainingPolicy_line10"):
            if self.per:
                batch, indices, weights = self.memory.sample(self.batch_size)
            else:
                batch = self.memory.sample(self.batch_size)
                weights = 1.

            policy_loss, entropies = self.calc_policy_loss(batch, weights) # added by tH 20210705
            update_params(self.policy_optim, self.policy, policy_loss, self.grad_clip)

            if self.entropy_tuning:
                entropy_loss = self.calc_entropy_loss(entropies, weights)
                update_params(self.alpha_optim, None, entropy_loss)
                self.alpha = self.log_alpha.exp()

    def calc_critic_4redq_loss(self, batch, weights):
        states, actions, rewards, next_states, dones = batch
        curr_qs = self.critic.allQs(states, actions)

        target_q = self.calc_target_q(*batch)

        # TD errors for updating priority weights
        errors = torch.abs(curr_qs[0].detach() - target_q) # TODO better to use average of all errors
        # We log means of Q to monitor training.
        mean_q1 = curr_qs[0].detach().mean().item()
        mean_q2 = curr_qs[1].detach().mean().item()

        # Critic loss is mean squared TD errors with priority weights.
        losses = []
        for curr_q in curr_qs:
            losses.append(torch.mean((curr_q - target_q).pow(2) * weights))
        return losses, errors, mean_q1, mean_q2

    def calc_critic_loss(self, batch, weights):
        assert self.method == "sac" or self.method == "duvn", "This method is only for sac or duvn method"

        curr_q1, curr_q2 = self.calc_current_q(*batch)
        target_q = self.calc_target_q(*batch)

        # TD errors for updating priority weights
        errors = torch.abs(curr_q1.detach() - target_q)
        # We log means of Q to monitor training.
        mean_q1 = curr_q1.detach().mean().item()
        mean_q2 = curr_q2.detach().mean().item()

        q1_loss = torch.mean((curr_q1 - target_q).pow(2) * weights)
        q2_loss = torch.mean((curr_q2 - target_q).pow(2) * weights)
        return q1_loss, q2_loss, errors, mean_q1, mean_q2

    def calc_policy_loss(self, batch, weights):
        states, actions, rewards, next_states, dones = batch

        # We re-sample actions to calculate expectations of Q.
        sampled_action, entropy, _ = self.policy.sample(states)
        # expectations of Q with clipped double Q technique
        if self.method == "redq":
            q = self.critic.averageQ(states, sampled_action)
        else:
            q1, q2 = self.critic(states, sampled_action)
            if self.method == "duvn":
                q2 = q1 # discard q2
            if self.target_drop_rate > 0.0:
                q = 0.5 * (q1 + q2)
            else:
                q = torch.min(q1, q2)
        # Policy objective is maximization of (Q + alpha * entropy) with
        # priority weights.
        policy_loss = torch.mean((- q - self.alpha * entropy) * weights)

        return policy_loss, entropy

    def calc_entropy_loss(self, entropy, weights):
        entropy_loss = -torch.mean(self.log_alpha * (self.target_entropy - entropy).detach() * weights)
        return entropy_loss

    def evaluate(self):
        episodes = self.eval_runs
        returns = np.zeros((episodes,), dtype=np.float32)

        # for return bias estimation TH
        sar_buf = [[] for _ in range(episodes) ] # episodes x (satte, action , reward)

        for i in range(episodes):
            state = self.env.reset()
            episode_reward = 0.
            done = False
            while not done:
                action = self.exploit(state)
                next_state, reward, done, _ = self.env.step(action)
                episode_reward += reward
                state = next_state
                # MCE store all (state, action, reward) TH 20210723
                sar_buf[i].append([state, action, reward])

            returns[i] = episode_reward

        mean_return = np.mean(returns)

        # calculate mean / std return bias. TH 20210801
        # - calculate MCE future discounted return (in backward)
        mc_discounted_return = [deque() for _ in range(episodes) ]
        for i in range(episodes):
            for re_tran in reversed(sar_buf[i]):
                if len(mc_discounted_return[i]) > 0:
                    mcret = re_tran[2] + self.gamma_n * mc_discounted_return[i][0]
                else:
                    mcret = re_tran[2]
                mc_discounted_return[i].appendleft(mcret)
        # - calculate normalized MCE return by averaging all MCE returns
        norm_coef = np.mean(list(itertools.chain.from_iterable(mc_discounted_return)))
        norm_coef = math.fabs(norm_coef) + 0.000001
        # - estimate return for all state action, and normalized score
        norm_scores = [[] for _ in range(episodes)]
        for i in range(episodes):
            # calculate normalized score
            states = np.array(sar_buf[i], dtype="object")[:, 0].tolist()
            actions = np.array(sar_buf[i], dtype="object")[:, 1].tolist()
            with torch.no_grad():
                state = torch.FloatTensor(states).to(self.device)
                action = torch.FloatTensor(actions).to(self.device)
                if self.method == "redq":
                    q = self.critic.averageQ(state, action)
                else:
                    q1, q2 = self.critic(state, action)
                    q = 0.5 * (q1 + q2)
                qs = q.to('cpu').numpy()
            for j in range(len(sar_buf[i])):
                score = (qs[j][0] - mc_discounted_return[i][j]) / norm_coef
                norm_scores[i].append(score)
        # calculate std
        flatten_norm_score = list(itertools.chain.from_iterable(norm_scores))
        mean_norm_score = np.mean(flatten_norm_score)
        std_norm_score = np.std(flatten_norm_score)
        print("mean norm score " + str(mean_norm_score))
        print("std norm score " + str(std_norm_score))

        self.writer.add_scalar(
            'reward/test', mean_return, self.steps)
        print('-' * 60)
        print(f'Num steps: {self.steps:<5}  '
              f'reward: {mean_return:<5.1f}')
        print('-' * 60)
        #
        with open(self.log_dir + "/" + "reward.csv", "a") as f:
            f.write(str(self.steps) + "," + str(mean_return) + ",\n")
            f.flush()
        #
        with open(self.log_dir + "/" + "avrbias.csv", "a") as f:
            f.write(str(self.steps) + "," + str(mean_norm_score) + ",\n")
            f.flush()
        #
        with open(self.log_dir + "/" + "stdbias.csv", "a") as f:
            f.write(str(self.steps) + "," + str(std_norm_score) + ",\n")
            f.flush()


    def save_models(self):
        self.policy.save(os.path.join(self.model_dir, 'policy.pth'))
        self.critic.save(os.path.join(self.model_dir, 'critic.pth'))
        self.critic_target.save(
            os.path.join(self.model_dir, 'critic_target.pth'))

    def __del__(self):
        self.writer.close()
        self.env.close()
