import random
import numpy as np

from gymnasium import spaces

from src.utils.misc import set_seed
from src.envs.half_cheetah import HalfCheetahEnv

CONSTANT_VELOCITY = 1.5

class HalfCheetahVelEnv(HalfCheetahEnv):
    """Half-cheetah environment with target velocity, as described in [1]. The
    code is adapted from
    https://github.com/cbfinn/maml_rl/blob/9c8e2ebd741cb0c7b8bf2d040c4caeeb8e06cc95/rllab/envs/mujoco/half_cheetah_env_rand.py

    The half-cheetah follows the dynamics from MuJoCo [2], and receives at each
    time step a reward composed of a control cost and a penalty equal to the
    difference between its current velocity and the target velocity. The tasks
    are generated by sampling the target velocities from the uniform
    distribution on [0, 2].

    [1] Chelsea Finn, Pieter Abbeel, Sergey Levine, "Model-Agnostic
        Meta-Learning for Fast Adaptation of Deep Networks", 2017
        (https://arxiv.org/abs/1703.03400)
    [2] Emanuel Todorov, Tom Erez, Yuval Tassa, "MuJoCo: A physics engine for
        model-based control", 2012
        (https://homes.cs.washington.edu/~todorov/papers/TodorovIROS12.pdf)
    """

    def __init__(self, goal_pos=None):
        if goal_pos is None:
            goal_pos = np.array(self.sample_tasks(1))
        self.set_task(goal_pos)
        self.task_dim = 1
        super(HalfCheetahVelEnv, self).__init__()
        self.observation_space = spaces.Box(low=-np.inf, high=np.inf, shape=(21,))

    def step(self, action):
        xposbefore = self.sim.data.qpos[0]
        self.do_simulation(action, self.frame_skip)
        xposafter = self.sim.data.qpos[0]

        forward_vel = (xposafter - xposbefore) / self.dt
        forward_reward = -1.0 * abs(forward_vel - self.goal_velocity)
        ctrl_cost = 0.5 * 1e-1 * np.sum(np.square(action))

        observation = self._get_obs()
        reward = forward_reward - ctrl_cost
        done = False
        infos = dict(reward_forward=forward_reward,
                     reward_ctrl=-ctrl_cost,
                     task=self.get_task(),
                     forward_vel=forward_vel)
        return observation, reward, done, done, infos

    def set_task(self, task):
        if isinstance(task, np.ndarray):
            task = task[0]
        self.goal_velocity = task
        self.goal = task

    def get_task(self):
        return np.array([self.goal_velocity])

    def sample_tasks(self, n_tasks):
        return [random.uniform(0.0, 3.0) for _ in range(n_tasks)]

    def reset_task(self, task):
        if task is None:
            task = self.sample_tasks(1)[0]
        self.set_task(task)
        self.reset()

    def reset(self, seed=0, options=None):
        obs, info = super().reset(seed=seed, options=options)
        # print("Obs space", self.observation_space)
        # print("Obs shape", obs.shape)
        return obs, info
        # return self.reset_model(), {}

    def pos_to_state(self, arg):
        return tuple(arg.tolist())


def train_test_goals_hcv(num_test_goals, seed):
    num_train_goals = 1000
    set_seed(seed)
    train_goals = np.array([[random.uniform(0.0, 3.0)] for _ in range(num_train_goals)])
    set_seed(seed + 1)
    test_goals = np.array([[random.uniform(0.0, 3.0)] for _ in range(num_test_goals)])
    test_goals_ood = np.array([[random.uniform(4.0, 7.0)] for _ in range(num_test_goals)])
    return train_goals, test_goals, test_goals_ood


if __name__ == "__main__":
    import gym
    gym.register(
        'HalfCheetahVel-v0',
        entry_point='src.envs.half_cheetah_vel:HalfCheetahVelEnv',
        max_episode_steps=200,
        kwargs={},
    )

    goal = np.array([1])
    env = gym.make("HalfCheetahVel-v0", goal_pos=goal)
    obs, _ = env.reset()
    print(obs)