import argparse
from itertools import count

import gym
import gym.spaces
import scipy.optimize
import numpy as np
import math

import torch
import torch.nn as nn
import torch.optim as optim
import torch.nn.functional as F
from models.old_models import *
from replay_memory import Memory
from running_state import ZFilter
from torch.autograd import Variable
from trpo import trpo_step
from utils import *
from loss import *
import time

import swimmer
import walker
import halfcheetah

import pickle

torch.utils.backcompat.broadcast_warning.enabled = True
torch.utils.backcompat.keepdim_warning.enabled = True

torch.set_default_tensor_type('torch.DoubleTensor')
use_cuda = torch.cuda.is_available()
device = torch.device("cuda" if use_cuda else "cpu")

parser = argparse.ArgumentParser(description='PyTorch actor-critic example')
parser.add_argument('--gamma', type=float, default=0.995, metavar='G',
                    help='discount factor (default: 0.995)')
parser.add_argument('--env-name', type=str, default="Reacher-v1", metavar='G',
                    help='name of the environment to run')
parser.add_argument('--tau', type=float, default=0.97, metavar='G',
                    help='gae (default: 0.97)')
parser.add_argument('--l2-reg', type=float, default=1e-3, metavar='G',
                    help='l2 regularization regression (default: 1e-3)')
parser.add_argument('--max-kl', type=float, default=1e-2, metavar='G',
                    help='max kl value (default: 1e-2)')
parser.add_argument('--damping', type=float, default=1e-1, metavar='G',
                    help='damping (default: 1e-1)')
parser.add_argument('--seed', type=int, default=1111, metavar='N',
                    help='random seed (default: 1111')
parser.add_argument('--batch-size', type=int, default=5000, metavar='N',
                    help='size of a single batch')
parser.add_argument('--log-interval', type=int, default=1, metavar='N',
                    help='interval between training status logs (default: 10)')
parser.add_argument('--eval-interval', type=int, default=1, metavar='N',
                    help='interval between training status logs (default: 10)')
parser.add_argument('--num-epochs', type=int, default=500, metavar='N',
                    help='number of epochs to train an expert')
parser.add_argument('--hidden-dim', type=int, default=64, metavar='H',
                    help='the size of hidden layers')
parser.add_argument('--lr', type=float, default=1e-3, metavar='L',
                    help='learning rate')
parser.add_argument('--vf-iters', type=int, default=30, metavar='V',
                    help='number of iterations of value function optimization iterations per each policy optimization step')
parser.add_argument('--vf-lr', type=float, default=3e-4, metavar='V',
                    help='learning rate of value network')
parser.add_argument('--render', action='store_true',
                    help='render the environment')
parser.add_argument('--xml', default=None, help='the xml configuration file')
parser.add_argument('--demo_files', nargs='+', help='the environment used for test')
parser.add_argument('--ratios', nargs='+', type=float, help='the ratio of demos to load')
parser.add_argument('--eval_epochs', type=int, default=10, help='the epochs for evaluation')
parser.add_argument('--save_path', help='the path to save model')
parser.add_argument('--feasibility_model', default=None, help='the path to the feasibility model')
parser.add_argument('--mode', help='the mode of feasibility')
parser.add_argument('--discount', type=float, default=0.9, help='the discount factor')
parser.add_argument('--distance_normalizer', type=float, default=5., help='the normalization factor for the distance')
args = parser.parse_args()

if args.seed == 1111:
    log_file = open('log/'+args.save_path.split('/')[-1].split('.pth')[0]+'.txt', 'w')
    save_path = args.save_path
else:
    log_file = open('log/'+args.save_path.split('/')[-1].split('.pth')[0]+'_seed_{}.txt'.format(args.seed), 'w')
    save_path = args.save_path.replace('.pth', '_seed_{}.pth'.format(args.seed))

env = gym.make(args.env_name, xml_file=args.xml, exclude_current_positions_from_observation=False)
f_env = gym.make(args.env_name, xml_file=args.xml, exclude_current_positions_from_observation=False)

num_inputs = env.observation_space.shape[0]
num_actions = env.action_space.shape[0]

def load_demos(demo_files, ratios):
    state_files = []
    trajs = []
    traj_traj_id = []
    traj_id = 0
    pair_traj_id = []
    init_obs = []
    for i in range(len(demo_files)):
        state_pairs = []
        demo_file = demo_files[i]
        raw_demos = pickle.load(open(demo_file, 'rb'))
        use_num = int(len(raw_demos['obs'])*ratios[i])
        current_state = raw_demos['obs'][0:use_num]
        next_state = raw_demos['next_obs'][0:use_num]
        trajs += [np.array(traj) for traj in current_state]
        if 'InvertedDoublePendulum' in str(type(env.env)):
            init_obs += raw_demos['init_obs']
        traj_traj_id += [i]*len(current_state)
        for j in range(len(current_state)):
            if 'Ant' in args.env_name:
                state_pairs.append(np.concatenate([np.array(current_state[j])[:,2:], np.array(next_state[j])[:,2:]], axis=1))
                pair_traj_id.append(np.array([traj_id]*np.array(current_state[j]).shape[0]))
            else:
                state_pairs.append(np.concatenate([np.array(current_state[j]), np.array(next_state[j])], axis=1))
                pair_traj_id.append(np.array([traj_id]*np.array(current_state[j]).shape[0]))
            traj_id += 1
        state_files.append(np.concatenate(state_pairs, axis=0))
    return state_files, trajs, np.concatenate(pair_traj_id, axis=0), np.array(traj_traj_id), init_obs


env.seed(args.seed)
torch.manual_seed(args.seed)
np.random.seed(args.seed)

def compute_feasibility_pair(expert_trajs, models, f_env):
    all_distance = []
    for index in range(len(expert_trajs)):
        expert_traj = expert_trajs[index]
        model = models[index]
        batch_size = 64
        batch_num = (expert_traj.shape[0]-1)//batch_size + 1
        with torch.no_grad():
            for i in range(batch_num):
                f_env.reset()
                action_mean, _, action_std = model(torch.from_numpy(expert_traj[i*batch_size:(i+1)*batch_size, 2:num_inputs]))
                action = torch.normal(action_mean, action_std).cpu().numpy()
                next_states = []
                for j in range(action_mean.shape[0]):
                    f_env.set_observation(expert_traj[i*batch_size+j])
                    next_state, _, _, _ = f_env.step(action[j])
                    next_states.append(next_state)
                next_states = np.array(next_states)
                distance = np.linalg.norm(expert_traj[i*batch_size:(i+1)*batch_size, num_inputs:] - next_states, ord=2, axis=1)
                all_distance.append(distance)
    all_distance = np.concatenate(all_distance, axis=0)
    feasibility = np.exp(-all_distance/3.)
    return feasibility

def compute_feasibility_traj(expert_trajs, traj_traj_id, models, f_env, init_obs):
    all_distance = []
    for index in range(len(expert_trajs)):
        if index >= 4:
            index = index % 2 + 2
        all_distance.append([])
        expert_traj = expert_trajs[index]
        model = models[traj_traj_id[index]]
        with torch.no_grad():
            f_env.reset()
            f_env.set_observation(expert_traj[0])
            state0 = expert_traj[0]
            state = expert_traj[0]
            for j in range(expert_traj.shape[0]-1):
                action_mean, _, action_std = model(torch.from_numpy(np.concatenate([state, state0], axis=0)).unsqueeze(0))
                action = action_mean.cpu().numpy()
                next_state, _, _, _ = f_env.step(action)
                state = next_state
                all_distance[-1].append(np.linalg.norm(expert_traj[j+1] - next_state, ord=2, axis=0)*(args.discount**j))
        all_distance[-1] = np.sum(all_distance[-1])
    all_distance = np.array(all_distance)
    all_distance = (all_distance + np.max(-all_distance))/args.distance_normalizer
    all_distance[all_distance>50] = 50.
    feasibility = np.exp(-all_distance)
    return feasibility

if args.feasibility_model is not None:
    if args.mode == 'pair':
        expert_pairs, _, _, _ = load_demos(args.demo_files, args.ratios)
    elif args.mode == 'traj':
        expert_pairs, expert_trajs, pair_traj_id, traj_traj_id, init_obs = load_demos(args.demo_files, args.ratios)
    feasibility_models = [Policy(num_inputs*2, num_actions, args.hidden_dim) for i in range(len(expert_pairs))]
    load_dict = torch.load(args.feasibility_model)
    for i in range(min(len(expert_pairs), 4)):
        feasibility_models[i].load_state_dict(load_dict['policy_'+str(i)])
    if args.mode == 'pair':
        feasibility = compute_feasibility_pair(expert_pairs, feasibility_models, f_env)
    elif args.mode == 'traj':
        feasibility_traj = compute_feasibility_traj(expert_trajs, traj_traj_id, feasibility_models, f_env, init_obs)
        feasibility = feasibility_traj[pair_traj_id]
else:
    expert_pairs, _, _, _, _ = load_demos(args.demo_files, args.ratios)
    feasibility = np.ones(sum([expert_traj.shape[0] for expert_traj in expert_pairs]))
expert_traj = np.concatenate(expert_pairs, axis=0)


policy_net = Policy(num_inputs, num_actions, args.hidden_dim)
value_net = Value(num_inputs, args.hidden_dim).to(device)
discriminator = Discriminator(num_inputs + num_inputs, args.hidden_dim).to(device)
disc_criterion = nn.BCEWithLogitsLoss()
value_criterion = nn.MSELoss()
disc_optimizer = optim.Adam(discriminator.parameters(), args.lr)
value_optimizer = optim.Adam(value_net.parameters(), args.vf_lr)

def select_action(state):
    state = torch.from_numpy(state).unsqueeze(0)
    action_mean, _, action_std = policy_net(Variable(state))
    action = torch.normal(action_mean, action_std)
    return action

def update_params(batch):
    rewards = torch.Tensor(batch.reward).to(device)
    masks = torch.Tensor(batch.mask).to(device)
    actions = torch.Tensor(np.concatenate(batch.action, 0)).to(device)
    states = torch.Tensor(batch.state).to(device)
    values = value_net(Variable(states))

    returns = torch.Tensor(actions.size(0),1).to(device)
    deltas = torch.Tensor(actions.size(0),1).to(device)
    advantages = torch.Tensor(actions.size(0),1).to(device)

    prev_return = 0
    prev_value = 0
    prev_advantage = 0
    for i in reversed(range(rewards.size(0))):
        returns[i] = rewards[i] + args.gamma * prev_return * masks[i]
        deltas[i] = rewards[i] + args.gamma * prev_value * masks[i] - values.data[i]
        advantages[i] = deltas[i] + args.gamma * args.tau * prev_advantage * masks[i]

        prev_return = returns[i, 0]
        prev_value = values.data[i, 0]
        prev_advantage = advantages[i, 0]

    targets = Variable(returns)

    batch_size = math.ceil(states.shape[0] / args.vf_iters)
    idx = np.random.permutation(states.shape[0])
    for i in range(args.vf_iters):
        smp_idx = idx[i * batch_size: (i + 1) * batch_size]
        smp_states = states[smp_idx, :]
        smp_targets = targets[smp_idx, :]
        
        value_optimizer.zero_grad()
        value_loss = value_criterion(value_net(Variable(smp_states)), smp_targets)
        value_loss.backward()
        value_optimizer.step()

    advantages = (advantages - advantages.mean()) / advantages.std()

    action_means, action_log_stds, action_stds = policy_net(Variable(states.cpu()))
    fixed_log_prob = normal_log_density(Variable(actions.cpu()), action_means, action_log_stds, action_stds).data.clone()

    def get_loss(volatile=None):
        action_means, action_log_stds, action_stds = policy_net(Variable(states.cpu()))
        log_prob = normal_log_density(Variable(actions.cpu()), action_means, action_log_stds, action_stds)
        action_loss = -Variable(advantages.cpu()) * torch.exp(log_prob - Variable(fixed_log_prob))
        return action_loss.mean()


    def get_kl():
        mean1, log_std1, std1 = policy_net(Variable(states.cpu()))

        mean0 = Variable(mean1.data)
        log_std0 = Variable(log_std1.data)
        std0 = Variable(std1.data)
        kl = log_std1 - log_std0 + (std0.pow(2) + (mean0 - mean1).pow(2)) / (2.0 * std1.pow(2)) - 0.5
        return kl.sum(1, keepdim=True)

    trpo_step(policy_net, get_loss, get_kl, args.max_kl, args.damping)

def expert_reward(states, actions):
    states = np.concatenate(states)
    actions = np.concatenate(actions)
    with torch.no_grad():
        state_action = torch.Tensor(np.concatenate([states, actions], 1)).to(device)
        return -F.logsigmoid(discriminator(state_action)).cpu().detach().numpy()


def evaluate(episode, best_reward, log_file):
  env.seed(1234)
  with torch.no_grad():
    avg_reward = 0.0
    for _ in range(args.eval_epochs):
        state = env.reset()
        for _ in range(10000): # Don't infinite loop while learning
            state = torch.from_numpy(state).unsqueeze(0)
            action, _, _ = policy_net(Variable(state))
            action = action.data[0].numpy()
            next_state, reward, done, _ = env.step(action)
            avg_reward += reward
            if done:
                break
            state = next_state
    print('Evaluation: Episode ', episode, ' Reward ', avg_reward / args.eval_epochs)
    log_file.write('Evaluation: Episode '+str(episode)+' Reward '+str(avg_reward / args.eval_epochs)+'\n')
    log_file.flush()
    if best_reward < avg_reward / args.eval_epochs:
        best_reward = avg_reward / args.eval_epochs
        torch.save({'policy':policy_net.state_dict(), 'value':value_net.state_dict(), 'discriminator':discriminator.state_dict(), 'disc_optimizer':disc_optimizer.state_dict(), 'rew':best_reward}, save_path)

all_idx = np.arange(0, expert_traj.shape[0])
p_idx = np.random.permutation(expert_traj.shape[0])
expert_traj = expert_traj[p_idx, :]
feasibility = feasibility[p_idx]

feasibility = feasibility / (np.sum(feasibility)+0.0000001)
feasibility[feasibility<(1./feasibility.shape[0])/10000000.] = 0
feasibility[0] = 1-np.sum(feasibility[1:])
print(feasibility[0:10])
best_reward = -1000000

for i_episode in range(args.num_epochs):
    env.seed(int(time.time()))
    memory = Memory()

    num_steps = 0
    num_episodes = 0
    
    reward_batch = []
    states = []
    actions = []
    next_states = []
    mem_actions = []
    mem_mask = []
    mem_next = []

    while num_steps < args.batch_size:
        state = env.reset()
   

        reward_sum = 0
        for t in range(10000): # Don't infinite loop while learning
            action = select_action(state)
            action = action.data[0].numpy()
            states.append(np.array([state]))
            actions.append(np.array([action]))
            next_state, true_reward, done, _ = env.step(action)
            next_states.append(np.array([next_state]))
            reward_sum += true_reward

            mask = 1
            if done:
                mask = 0

            mem_mask.append(mask)
            mem_next.append(next_state)

            if done:
                break

            state = next_state
        num_steps += (t-1)
        num_episodes += 1

        reward_batch.append(reward_sum)

    if i_episode % args.eval_interval == 0:
        evaluate(i_episode, best_reward, log_file)

    rewards = expert_reward(states, next_states)
    for idx in range(len(states)):
        memory.push(states[idx][0], actions[idx], mem_mask[idx], mem_next[idx], \
                    rewards[idx][0])
    batch = memory.sample()
    update_params(batch)

    ### update discriminator ###
    next_states = torch.from_numpy(np.concatenate(next_states))
    states = torch.from_numpy(np.concatenate(states))
   

    labeled_num = min(expert_traj.shape[0], num_steps)

    idx = np.random.choice(all_idx, labeled_num, p=feasibility.reshape(-1))

    expert_state_action = expert_traj[idx, :]
    expert_state_action = torch.Tensor(expert_state_action).to(device)
    real = discriminator(expert_state_action)    

    state_action = torch.cat((states, next_states), 1).to(device)
    fake = discriminator(state_action)

    disc_optimizer.zero_grad()
    disc_loss = disc_criterion(fake, torch.ones(fake.size(0), 1).to(device)) + \
                disc_criterion(real, torch.zeros(real.size(0), 1).to(device))
    
    
    disc_loss.backward()
    disc_optimizer.step()
    ############################

    if i_episode % args.log_interval == 0:
        print('Episode {}\tAverage reward: {:.2f}\tMax reward: {:.2f}\tLoss (disc): {:.2f}'.format(i_episode, np.mean(reward_batch), max(reward_batch), disc_loss.item()))
        log_file.write('Episode {}\tAverage reward: {:.2f}\tMax reward: {:.2f}\tLoss (disc): {:.2f}\n'.format(i_episode, np.mean(reward_batch), max(reward_batch), disc_loss.item()))
        log_file.flush()
