# SPDX-FileCopyrightText: Copyright (c) 2021 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
# SPDX-License-Identifier: BSD-3-Clause
# 
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice, this
# list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# 3. Neither the name of the copyright holder nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#
# Copyright (c) 2021 ETH Zurich, Nikita Rudin

import time
import os
from collections import deque
import statistics

# from torch.utils.tensorboard import SummaryWriter
import torch
import torch.optim as optim
import wandb
# import ml_runlog
import datetime
import torch
from collections import deque
import random
import numpy as np


from rsl_rl.algorithms import PPO
from rsl_rl.modules import *
from rsl_rl.env import VecEnv
import sys
from copy import copy, deepcopy
import warnings
from rsl_rl.utils.utils import Normalizer

import torch.nn as nn
import torch.optim as optim

from rsl_rl.storage import RolloutStorage, ReplayBuffer
import torch.nn.functional as F



# Replay Buffer
class ReplayBuffer_selector:
    def __init__(self, capacity, device):
        self.buffer = deque(maxlen=capacity)
        self.device = device

    # def add(self, state, action, reward, next_state, done):
        # self.buffer.append((state, action, reward, next_state, done))
    
    def add(self, state, action, reward, next_state, done):
        # self.buffer.append((
        #     np.array(state), 
        #     np.array(action), 
        #     np.array(reward), 
        #     np.array(next_state), 
        #     np.array(done)
        # ))
        self.buffer.append((
            np.asarray(state), 
            np.asarray(action), 
            np.asarray(reward), 
            np.asarray(next_state), 
            np.asarray(done)
        ))

    def sample(self, batch_size):
        # 从buffer中随机采样
        indices = np.random.choice(len(self.buffer), batch_size, replace=False)
        sampled_data = [self.buffer[i] for i in indices]

        # 解包数据
        # states, actions, rewards, next_states, dones = zip(*sampled_data)
        
        # 转换为 NumPy 数组
        states, actions, rewards, next_states, dones = map(np.array, zip(*sampled_data))


        # 转换为张量，确保形状正确
        return (
            torch.tensor(states).to(self.device),  # 确保为二维数组
            torch.tensor(actions).to(self.device),             # 动作为一维数组
            torch.tensor(rewards).to(self.device),          # 奖励为一维数组
            torch.tensor(next_states).to(self.device),  # 确保为二维数组
            torch.tensor(dones).to(self.device),            # 完成标志为一维数组
        )


    def __len__(self):
        return len(self.buffer)


# Reward Normalizer
class RewardNormalizer:
    def __init__(self, gamma=0.99):
        self.gamma = gamma
        self.running_mean = 0
        self.var = 1
        self.count = 1e-4

    def normalize(self, reward):
        self.running_mean = self.gamma * self.running_mean + (1 - self.gamma) * reward
        self.var = self.gamma * self.var + (1 - self.gamma) * (reward - self.running_mean) ** 2
        self.count += 1
        std = (self.var / self.count) ** 0.5
        return (reward - self.running_mean) / (std + 1e-8)

class SelectorNetwork(nn.Module):
    def __init__(self, input_dim, output_dim=2, dropout_prob=0.1):
        super(SelectorNetwork, self).__init__()
        # 定义网络层
        self.fc1 = nn.Linear(input_dim, 256)
        self.fc2 = nn.Linear(256, 128)
        self.fc3 = nn.Linear(128, 64)
        self.fc4 = nn.Linear(64, output_dim)  # 输出 Q 值，每个动作一个
        
        self.dropout = nn.Dropout(p=dropout_prob)
        
        # 权重初始化
        self._init_weights()

    def _init_weights(self):
        """权重初始化"""
        nn.init.xavier_uniform_(self.fc1.weight)
        nn.init.zeros_(self.fc1.bias)
        nn.init.xavier_uniform_(self.fc2.weight)
        nn.init.zeros_(self.fc2.bias)
        nn.init.xavier_uniform_(self.fc3.weight)
        nn.init.zeros_(self.fc3.bias)
        nn.init.xavier_uniform_(self.fc4.weight)
        nn.init.zeros_(self.fc4.bias)


    def forward(self, x):
        # 前向传播
        x = F.relu(self.fc1(x))  # 第一层，带 LayerNorm 和 ReLU
        x = self.dropout(x)
        x = F.relu(self.fc2(x))  # 第二层，带 LayerNorm 和 ReLU
        x = F.relu(self.fc3(x))  # 第三层，带 LayerNorm 和 ReLU
        x = self.fc4(x)                    # 输出层，不加激活函数
        return x


# RL Trainer
class Selector_Trainer:
    def __init__(self,
            env: VecEnv,
            train_cfg,
            log_dir=None,
            init_wandb=True,
            device='cpu', **kwargs):

        self.env = env
        self.device = env.device
        self.batch_size = 128
        self.lr = 1e-4
        self.gamma = 0.99
        self.max_grad_norm = 1.0
        self.replay_buffer = ReplayBuffer_selector(capacity=2000000, device=self.device)
        self.reward_normalizer = RewardNormalizer(gamma=self.gamma)

        self.prev_action = None  # 用于记录上一次的选择
        self.switch_penalty = 0.0001  # 定义切换惩罚的系数    0.01

        # Selector network and optimizer
        self.policy_cfg = train_cfg["policy"]
        self.estimator_cfg = train_cfg["estimator"]

        # obs_dim = self.policy_cfg['selector_input'] + 1
        obs_dim = self.policy_cfg['selector_input']
        self.selector = SelectorNetwork(input_dim=obs_dim).to(self.device)
        self.target_selector = deepcopy(self.selector).to(self.device)      # 目标网络
        self.target_selector.eval()  # 目标网络只推理，不训练
        self.target_update_frequency = 50  # 目标网络更新频率（单位：步数）

        self.optimizer = optim.Adam(self.selector.parameters(), lr=self.lr)
        self.loss_fn = nn.MSELoss()  
        
        # ε-greedy 参数
        self.epsilon = 0.1  # 初始 ε
        self.epsilon_min = 0.0002  # 最小 ε  
        self.epsilon_decay = 0.998  # 衰减率  0.999


        root_path = "/data1/selector_policy/test/nips/0.0001_2"
        self.model_save_path = "/data1/selector_policy/test/nips/0.0001_2"

        # loco_name = "walk_locomotion_test8-19800-actor_jit.pt"
        # reco_name = "walk_recovery_test8-19800-actor_jit.pt"

        # loco_name = "walk_ours3-39800-actor_jit.pt"
        # reco_name = "walk_recovery_no_case3-39800-actor_jit.pt"

        # loco_name = "walk_ours_real4-42000-actor_jit.pt"
        # reco_name = "walk_recovery_real4-44000-actor_jit.pt"

        loco_name = "goal_tracking-34800-actor_jit.pt"
        # reco_name = "recovery_demo3-19800-actor_jit.pt"
        reco_name = "recovery-34800-actor_jit.pt"
        


        loco_path = os.path.join(root_path, loco_name)
        reco_path = os.path.join(root_path, reco_name)
        print("current device is:", self.device)

        self.locomotion_policy = torch.jit.load(loco_path, map_location=self.device).to(self.device)
        self.recovery_policy =  torch.jit.load(reco_path, map_location=self.device).to(self.device)

        print("policy", self.locomotion_policy )


# Double-DQN
    def update_selector(self, update_step):
        if len(self.replay_buffer) < self.batch_size:
            return

        # Sample from the replay buffer
        states, actions, rewards, next_states, dones = self.replay_buffer.sample(self.batch_size)
        # print('states', states.shape)
        # Compute Q(s, a) using main network
        q_values = self.selector(states)  # 主网络输出 Q 值
        q_values = q_values.gather(1, actions.unsqueeze(-1).long()).squeeze(-1)  # 提取选择动作的 Q 值

        # Compute Q_target using target network
        with torch.no_grad():
            # 主网络选择下一状态的动作
            next_q_values_online = self.selector(next_states)
            max_action_indices = next_q_values_online.argmax(dim=1)
            # 目标网络评估所选动作的Q值
            next_q_values_target = self.target_selector(next_states)
            q_targets = rewards + self.gamma * next_q_values_target.gather(1, max_action_indices.unsqueeze(-1)).squeeze(-1) * (1 - dones.float())

        # Compute loss
        loss = self.loss_fn(q_values, q_targets)

        # Backward pass and optimizer step
        self.optimizer.zero_grad()
        loss.backward()
        nn.utils.clip_grad_norm_(self.selector.parameters(), 1.0)  # 梯度裁剪
        self.optimizer.step()

        # Update target network every few steps
        if update_step % self.target_update_frequency == 0:
            self.target_selector.load_state_dict(self.selector.state_dict())  # 同步参数

        return loss

    def select_action(self, q_values):
        # ε-greedy 策略
        if random.random() < self.epsilon:
            # print(torch.randint(0, 2, (q_values.shape[0]), device=self.device).long().shape)
            # print('random', torch.randint(0, 2, (q_values.shape[0],), device=self.device).long())
            return torch.randint(0, 2, (q_values.shape[0],), device=self.device).long()
        else:
            # print(q_values.argmax(dim=1).long().shape)
            return q_values.argmax(dim=1).long()  # 选择具有最大 Q 值的动作


    def save_selector(self, filename):
        """将 Selector Network 保存为 state_dict 格式"""
        # 获取模型的状态字典
        model_state_dict = self.selector.state_dict()
        
        # 保存为普通的字典格式
        filename = os.path.join(self.model_save_path, filename)
        torch.save(model_state_dict, filename)


    def load_selector(self, filename):
        self.selector = torch.load(self.model_save_path, filename).to(self.device)



    def learn(self, num_learning_iterations, num_steps_per_env, init_at_random_ep_len=False):
        if init_at_random_ep_len:
            self.env.episode_length_buf = torch.randint_like(
                self.env.episode_length_buf, high=int(self.env.max_episode_length)
            )

        cur_reward_sum = torch.zeros(self.env.num_envs, dtype=torch.float, device=self.device)
        cur_episode_length = torch.zeros(self.env.num_envs, dtype=torch.float, device=self.device)
        cur_switch_penalty_sum = torch.zeros(self.env.num_envs, dtype=torch.float, device=self.device)

        rewbuffer = deque(maxlen=100)
        lenbuffer = deque(maxlen=100)
        switch_penalty_buf = deque(maxlen=100)
        
        # selected_action_history = torch.zeros(self.env.num_envs, 10 , device=self.device)

        frequence_period = 100

        locomotion_sum = torch.zeros(self.env.num_envs, dtype=torch.float, device=self.device)
        recovery_sum = torch.zeros(self.env.num_envs, dtype=torch.float, device=self.device)

        obs = self.env.get_observations()
        print(obs.shape)
        # print(self.device)
        self.locomotion_policy 
        z, v = self.locomotion_policy.estimator(obs.detach()[:, self.estimator_cfg['prop_start'] - 9 * self.estimator_cfg['prop_dim'] : self.estimator_cfg['prop_start'] + self.estimator_cfg['prop_dim']])
        latent = torch.cat([z,v],dim = 1)

        obs = torch.cat([obs[:, :self.estimator_cfg['prop_start'] + self.estimator_cfg['prop_dim']], latent],dim = 1)
        self.prev_action = torch.zeros(self.env.num_envs, device=self.device)
        # obs_h = torch.cat((obs, self.prev_action.unsqueeze(1)), dim=-1)

        ep_infos = []

        for it in range(num_learning_iterations):
            locomotion_percentages = []  # 存储每个 episode 中 locomotion 策略选择的百分比
            with torch.inference_mode():
                for step in range(num_steps_per_env):
                    # obs_h = torch.cat((obs, selected_action_history), dim = -1)
                    with torch.no_grad():
                        # Q-values for both actions (0: locomotion, 1self: recovery)
                        q_values = self.selector(obs)
                        # q_values = self.selector(obs_h)
                        selected_action = self.select_action(q_values)  # 使用 ε-greedy 选择动作

                    # 更新selected_action时计算切换惩罚

                    switch_penalty = - (self.prev_action != selected_action).float() * self.switch_penalty
                    # print("self.prev_action", self.prev_action.shape)  # [env_nums]

                    # Execute the selected policy for all environments
                    actions = torch.empty((obs.shape[0], 19), device=self.device)  # Initialize action tensor

                    locomotion_mask = selected_action == 0  # Mask for locomotion policy
                    recovery_mask = selected_action == 1  # Mask for recovery policy

                    # Compute actions for each policy
                    if locomotion_mask.any():
                        actions[locomotion_mask] = self.locomotion_policy(obs[locomotion_mask])
                        locomotion_sum[locomotion_mask] +=1
                    if recovery_mask.any():
                        actions[recovery_mask] = self.recovery_policy(obs[recovery_mask])
                        recovery_sum[recovery_mask] +=1

                    # Interact with the environment
                    next_obs, privileged_obs, reward, dones, info = self.env.step(actions.to(self.device))
                    # next_obs = torch.tensor(next_obs, dtype=torch.float32).to(self.device)

                    # Compute estimation and latent
                    if locomotion_mask.any():
                        z, v = self.locomotion_policy.estimator(next_obs.detach()[:, self.estimator_cfg['prop_start'] - 9 * self.estimator_cfg['prop_dim'] : self.estimator_cfg['prop_start'] + self.estimator_cfg['prop_dim']])
                        loco_latent = torch.cat([z,v],dim = 1)
                        latent[locomotion_mask] = loco_latent[locomotion_mask]
                    if recovery_mask.any():
                        z, v = self.recovery_policy.estimator(next_obs.detach()[:, self.estimator_cfg['prop_start'] - 9 * self.estimator_cfg['prop_dim'] : self.estimator_cfg['prop_start'] + self.estimator_cfg['prop_dim']])
                        reco_latent = torch.cat([z,v],dim = 1)
                        latent[recovery_mask] = reco_latent[recovery_mask]

                    next_obs = torch.cat([next_obs[:, :self.estimator_cfg['prop_start'] + self.estimator_cfg['prop_dim']], latent],dim = 1)
                    # next_obs_h = torch.cat((next_obs, self.prev_action.unsqueeze(1)), dim = -1)

                    normalized_reward = reward + switch_penalty

                    if 'episode' in info:
                        ep_infos.append(info['episode'])

                    # print('obs', obs.shape)
                    cur_reward_sum += normalized_reward
                    cur_switch_penalty_sum += switch_penalty
                    cur_episode_length +=1
                    new_ids = (dones > 0).nonzero(as_tuple=False)

                    rewbuffer.extend(cur_reward_sum[new_ids][:, 0].cpu().numpy().tolist())
                    lenbuffer.extend(cur_episode_length[new_ids][:, 0].cpu().numpy().tolist())
                    switch_penalty_buf.extend(cur_switch_penalty_sum[new_ids][:, 0].cpu().numpy().tolist())

                    cur_reward_sum[new_ids] = 0
                    cur_switch_penalty_sum[new_ids] = 0
                    cur_episode_length[new_ids] = 0
                    locomotion_sum[new_ids] = 0
                    recovery_sum[new_ids] = 0

                    for i in range(obs.shape[0]):  # 遍历每个环境的状态
                        self.replay_buffer.add(
                            obs[i].detach().cpu(),                  # 单个环境的状态
                            selected_action[i].detach().cpu(),      # 对应的动作
                            normalized_reward[i].detach().cpu(),               # 对应的奖励
                            next_obs[i].detach().cpu(),             # 下一状态
                            dones[i].detach().cpu()                 # 完成标志
                        )
                    
                    obs = next_obs
                    # obs_h = next_obs_h
                    # 更新prev_action
                    self.prev_action = selected_action.clone()


            loco_percentage = (locomotion_sum / (locomotion_sum + recovery_sum + 1e-6)).mean()
            reco_percentage = (recovery_sum / (locomotion_sum + recovery_sum + 1e-6)).mean()

            # Update selector
            for i in range(2):
                dqn_loss = self.update_selector(update_step = it)

            # 衰减 ε
            if self.epsilon > self.epsilon_min:
                self.epsilon *= self.epsilon_decay


            # Logging
            # 打印本次迭代的 locomotion 策略选择的平均百分比
            # print(dqn_loss)
            
            print(f"Iteration {it}, Avg Locomotion Percentage: {loco_percentage * 100:.2f}%")
            print(f"Iteration {it}, Avg recovery Percentage: {reco_percentage * 100:.2f}%")
            print(f"Iteration {it}, Replay Buffer Size: {len(self.replay_buffer)}")
            print('Current mean reward', cur_reward_sum.mean())

            print('Current mean episode length', cur_episode_length.mean())
            print(f"Q-Learning Loss: {dqn_loss.item()}")


            locs = locals()
            wandb_dict = {
                "Iteration": it,
                "DQN_Loss": dqn_loss.item(),
                "Locomotion_Percentage": loco_percentage.item(),
                "Recovery_Percentage": reco_percentage.item(),
                "Replay_Buffer_Size": len(self.replay_buffer),
                "Mean_Reward": torch.mean(torch.tensor(rewbuffer, dtype=torch.float32, device=self.device)).item(),
                "Mean_Episode_Length": torch.mean(torch.tensor(lenbuffer, dtype=torch.float32, device=self.device)).item(),
                "Switch_penalty": torch.mean(torch.tensor(switch_penalty_buf, dtype=torch.float32, device=self.device)).item(),
                'Epsilon': self.epsilon
            }

            if locs.get('ep_infos') and len(locs['ep_infos']) > 0:  # 确保 ep_infos 存在且非空
                for key in locs['ep_infos'][0]:
                    infotensor = torch.empty(0, device=self.device)  # 初始化空张量
                    for ep_info in locs['ep_infos']:
                        # 将 ep_info[key] 转换为张量，处理标量和零维张量情况
                        value_tensor = torch.as_tensor(ep_info[key], device=self.device, dtype=torch.float32).flatten()
                        infotensor = torch.cat((infotensor, value_tensor))  # 拼接张量
                    value = infotensor.mean()  # 计算均值
                    # 根据 key 名称分类并记录到 wandb_dict
                    if "tracking" in key:
                        wandb_dict[f"Episode_rew_tracking/{key}"] = value
                    elif "curriculum" in key:
                        wandb_dict[f"Episode_curriculum/{key}"] = value
                    elif "terrain_level" in key:
                        wandb_dict[f"Episode_terrain_level/{key}"] = value
                    else:
                        wandb_dict[f"Episode_rew_regularization/{key}"] = value
    
            wandb.log(wandb_dict)


            # 每个学习迭代结束后保存 selector 模型
            if it < 2000:
                if (it + 1) % 500 == 0:  # 每10次迭代保存一次模型
                    # path = "/data1/selector_policy"
                    selector_filename = f"selector_model_{it + 1}.pt"
                    self.save_selector(selector_filename)
                    # self.load_selector(selector_filename)  # 加载最新的模型
                    print(f"Selector model saved to {selector_filename}")
            else:
                if (it + 1) % 200 == 0:  # 每10次迭代保存一次模型
                    # path = "/data1/selector_policy"
                    selector_filename = f"selector_model_{it + 1}.pt"
                    self.save_selector(selector_filename)
                    # self.load_selector(selector_filename)  # 加载最新的模型
                    print(f"Selector model saved to {selector_filename}")      
            ep_infos.clear()