"""Sawyer environment to pickup and insert pegs.

The observation is configured to included the gripper position, gripper distance and head of the object.
The sparse reward function only considers the position of the object, not the position of the gripper.
"""
import os

import numpy as np
from gym.spaces import Box

from metaworld.envs import reward_utils
from metaworld.envs.mujoco.sawyer_xyz.sawyer_xyz_env import SawyerXYZEnv, _assert_task_is_set
from scipy.spatial.transform import Rotation

import mujoco_py
import numpy as np

initial_states = np.array(
      [[ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.00313463,
         0.68326396,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        , -0.04035005,
         0.67949003,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.02531051,
         0.6074387 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.05957219,
         0.6271171 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        , -0.07566337,
         0.62575287,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        , -0.01177235,
         0.55206996,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.02779735,
         0.54707706,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.01835314,
         0.5329686 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.02690855,
         0.6263067 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.01766127,
         0.59630984,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.0560186 ,
         0.6634998 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        , -0.03950658,
         0.6323736 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        , -0.03216827,
         0.5247563 ,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.01265727,
         0.69466716,  0.02      ],
       [ 0.00615235,  0.6001898 ,  0.19430117,  1.        ,  0.05076993,
         0.6025737 ,  0.02      ]])

# goal_states = np.array([[0.0, 0.6, 0.2, 1.0, -0.3 + 0.03, 0.6, 0.0 + 0.13]])


initial_ee_pos = np.array([-8.49578129e-07,  6.00251578e-01,  1.96188105e-01])

# Same as SawyerPeg but without box
class SawyerPegPickAndPlaceV2(SawyerXYZEnv):
  max_path_length = int(1e8)
  TARGET_RADIUS = 0.05

  def __init__(self, reward_type='dense', full_state_goal = False, reset_at_goal=False):

    hand_low = (-0.5, 0.20, 0.05)
    hand_high = (0.5, 1, 0.5)
    obj_low = (-0.4, 0.2, 0.01478)
    obj_high = (0.4, 1.0, 0.4)
    goal_low = (-0.6, 0.2, 0.01478)
    goal_high = (0.6, 1.0, 0.4)



    super().__init__(
        self.model_name,
        hand_low=hand_low,
        hand_high=hand_high,
    )
    self.random_init = False 
    self.full_state_goal = full_state_goal
    self.init_config = {
        'obj_init_pos': np.array([0.4, 0.8, 0.02]),
        'hand_init_pos': np.array([0.4, 0.8, 0.2]),
    }

    self.initial_states = initial_states
    if full_state_goal:
      goal_states = np.array([[0.4, 0.8, 0.2, 1.0, -0.3, 0.4, 0.2]])
    else:
      goal_states = np.array([[-0.3, 0.4, 0.2]])
    self.goal_states = goal_states

    self.obj_init_pos = self.init_config['obj_init_pos']
    self.hand_init_pos = self.init_config['hand_init_pos']
    self.peg_head_pos_init = self._get_site_pos('pegHead')
    self.goal = goal_states[0]
    if full_state_goal:
      self._target_pos = self.goal[4:].copy()
    else:
      self._target_pos = self.goal.copy()


    self._partially_observable = False
    self._set_task_called = True
    self._freeze_rand_vec = False

    self._reset_at_goal = reset_at_goal
    if reset_at_goal:
      raise NotImplementedError
    self._reward_type = reward_type
    
    self._random_reset_space = Box(
      np.hstack((obj_low, goal_low)),
      np.hstack((obj_high, goal_high)),
    )
    self.goal_space = Box(
      np.array(goal_low), # + np.array([.03, .0, .0]),
      np.array(goal_high) # + np.array([.03, .0, .0])
    )

    # self.obj_goal_space = Box(
    #   np.array(obj_goal_low),
    #   np.array(obj_goal_high)
    # )
    self.obj_goal_height_offset = 0.01478

    
    self.metadata = {
      'render.modes': ['human', 'rgb_array'],
      'video.frames_per_second': int(np.round(1.0 / self.dt))
    }

    self.initial_ee_pos = initial_ee_pos.copy()
  @property
  def model_name(self):
    return os.path.join(
      os.path.dirname(os.path.realpath(__file__)), "metaworld_assets/sawyer_xyz", 'sawyer_peg_pick_and_place.xml')

  @property
  def observation_space(self):
    obj_low = np.full(3, -np.inf)
    obj_high = np.full(3, +np.inf)
    goal_low = self.goal_space.low
    goal_high = self.goal_space.high
    gripper_low = -1
    gripper_high = 1
    if self.full_state_goal:
      return Box(
        np.hstack((self._HAND_SPACE.low, gripper_low, obj_low, self._HAND_SPACE.low, gripper_low, goal_low)),
        np.hstack((self._HAND_SPACE.high, gripper_high, obj_high, self._HAND_SPACE.high, gripper_high, goal_high))
      )
    else:
      return Box(
        np.hstack((self._HAND_SPACE.low, gripper_low, obj_low, goal_low)),
        np.hstack((self._HAND_SPACE.high, gripper_high, obj_high, goal_high))
      )

  def _get_obs(self):
    obs = super()._get_obs()
    # xyz and gripper distance for end effector
    endeff_config = obs[:4]
    obj_pos = self._get_pos_objects()
    obs = np.concatenate([
        endeff_config, obj_pos, self.goal,
    ])
    return obs

  def get_next_goal(self):
    if not self._reset_at_goal:
      return self.goal_states[0].copy()

    else:
      raise NotImplementedError
      num_goals = self.initial_states.shape[0]
      goal_idx = np.random.randint(0, num_goals)
      return self.initial_states[goal_idx].copy()
  

  def reset(self, goal=None):
    self.curr_path_length = 0
    self._did_see_sim_exception = False
    self.sim.reset()
    ob = self.reset_model(goal = goal)
    if self.viewer is not None:
        self.viewer_setup()
    
    return ob

  def reset_goal(self, goal=None, add_noise = False):
    if goal is None:
      goal = self.get_next_goal()
    
    if add_noise: 
      goal = goal + np.random.normal(scale=0.002, size = goal.shape)

    self.goal = goal
    if self.full_state_goal:
      self._target_gripper_distance = goal[3]
      self._target_pos = goal[4:]
      self._end_effector_goal = goal[:3]
    else:
      self._target_pos = goal.copy()

    self.sim.model.site_pos[self.model.site_name2id('goal')] = self._target_pos

  @_assert_task_is_set
  def evaluate_state(self, obs, action):
    obj = obs[4:7]

    reward, tcp_to_obj, tcp_open, obj_to_target, grasp_reward, in_place_reward, collision_box_front, ip_orig, is_success = self.compute_reward(obs, action)
    grasp_success = float(tcp_to_obj < 0.02 and (tcp_open > 0) and (obj[2] - 0.01 > self.obj_init_pos[2]))
    success = float(obj_to_target <= self.TARGET_RADIUS)
    near_object = float(tcp_to_obj <= 0.03)

    info = {
        'success': success,
        'near_object': near_object,
        'grasp_success': grasp_success,
        'grasp_reward': grasp_reward,
        'in_place_reward': in_place_reward,
        'obj_to_target': obj_to_target,
        'unscaled_reward': reward,
        'is_success' : is_success,
    }

    return reward, info

  def _get_pos_objects(self):
    return self._get_site_pos('pegHead')

  def _get_quat_objects(self):
    return Rotation.from_matrix(self.data.get_site_xmat('pegHead')).as_quat()

  def reset_model(self, goal=None):
    if not self._reset_at_goal:
      self._reset_hand()
      self.reset_goal(goal=goal)
      
      pos_peg = self.obj_init_pos
      if self.random_init:
        raise NotImplementedError

      self.obj_init_pos = pos_peg
      self._set_obj_xyz(self.obj_init_pos + np.concatenate([np.random.uniform(-0.05, 0.05, size=2), np.zeros([1])], axis=-1))
      self.peg_head_pos_init = self._get_site_pos('pegHead')
    else:
      raise NotImplementedError

    return self._get_obs()

  def compute_reward(self, obs, action=None):
    tcp = obs[:3]
    obj = obs[4:7] - self._get_site_pos('pegHead') \
                   + self._get_site_pos('pegGrasp')
    obj_head = obs[4:7] 
    tcp_opened = obs[3]
    if self.full_state_goal: # actually same both if & else
      target = obs[11:14]
    else:
      target = obs[-3:]

    tcp_to_obj = np.linalg.norm(obj - tcp)
    
    scale = np.array([1., 2., 2.])
    #  force agent to pick up object then insert
    obj_to_target = np.linalg.norm((obj_head - target) * scale)

    in_place_margin = np.linalg.norm((self.peg_head_pos_init - target) * scale)
    in_place = reward_utils.tolerance(obj_to_target,
                                bounds=(0, self.TARGET_RADIUS),
                                margin=in_place_margin,
                                sigmoid='long_tail',)
    ip_orig = in_place
    # brc_col_box_1 = self._get_site_pos('bottom_right_corner_collision_box_1')
    # tlc_col_box_1 = self._get_site_pos('top_left_corner_collision_box_1')

    # brc_col_box_2 = self._get_site_pos('bottom_right_corner_collision_box_2')
    # tlc_col_box_2 = self._get_site_pos('top_left_corner_collision_box_2')
    # collision_box_bottom_1 = reward_utils.rect_prism_tolerance(curr=obj_head,
    #                                                            one=tlc_col_box_1,
    #                                                            zero=brc_col_box_1)
    # collision_box_bottom_2 = reward_utils.rect_prism_tolerance(curr=obj_head,
    #                                                            one=tlc_col_box_2,
    #                                                            zero=brc_col_box_2)
    # collision_boxes = reward_utils.hamacher_product(collision_box_bottom_2,
    #                                                 collision_box_bottom_1)
    # in_place = reward_utils.hamacher_product(in_place,
    #                                          collision_boxes)
    
    collision_boxes = None

    if tcp_to_obj < 0.08 and (tcp_opened > 0) and (obj[2] - 0.01 > self.obj_init_pos[2]):
      object_grasped = 1.

    # this is the only part that requires the action, only with dense reward
    elif self._reward_type == 'dense':
      pad_success_margin = 0.03
      object_reach_radius=0.01
      x_z_margin = 0.005
      obj_radius = 0.0075
      object_grasped = self._gripper_caging_reward(action,
                                                   obj,
                                                   object_reach_radius=object_reach_radius,
                                                   obj_radius=obj_radius,
                                                   pad_success_thresh=pad_success_margin,
                                                   xz_thresh=x_z_margin,
                                                   high_density=True)
    elif self._reward_type == 'sparse':
      object_grasped = 0.



    if self._reward_type == 'sparse':
      reward = (self.is_successful(obs=obs)).astype(np.float)
    
    is_success = self.is_successful(obs=obs)

    return [reward, tcp_to_obj, tcp_opened, obj_to_target, object_grasped, in_place, collision_boxes, ip_orig, is_success]

  def is_successful(self, obs=None):
    if obs is None:
      obs = self._get_obs()


    if self.full_state_goal: # actually same both if & else
      return np.linalg.norm(obs[..., 4:7] - obs[..., 11:14], axis =-1) <= self.TARGET_RADIUS
    else:
      return np.linalg.norm(obs[..., 4:7] - obs[..., -3:], axis =-1) <= self.TARGET_RADIUS

  def viewer_setup(self):
    self.viewer.cam.distance = 2.0
    self.viewer.cam.elevation = -20
    self.viewer.cam.azimuth = -150

  def _get_viewer(self, mode):
    self.viewer = self._viewers.get(mode)
    if self.viewer is None:
      if mode == 'human':
        self.viewer = mujoco_py.MjViewer(self.sim)
      if 'rgb_array' in mode:
        self.viewer = mujoco_py.MjRenderContextOffscreen(self.sim)
      self._viewers[mode] = self.viewer

    self.viewer_setup()
    return self.viewer
  
  def render(self, mode='human', camera_name='topview'):
    if mode == 'human':
      self._get_viewer(mode).render()
    elif mode == 'rgb_array':
      return self.sim.render(
          640, 480,
          mode='offscreen',
          camera_name=camera_name
      ) 
    else:
      raise ValueError("mode can only be either 'human' or 'rgb_array'")

  def close(self):
    if self.viewer is not None:
      if isinstance(self.viewer, mujoco_py.MjViewer):
        glfw.destroy_window(self.viewer.window)
      self.viewer = None
  
  def set_tstar_states(self, hand_pos=None, obj_pos=None):
      if hand_pos is not None:
        for i in range(hand_pos.shape[0]):
          self.sim.model.site_pos[self.model.site_name2id('hand_'+str(i+1))] = hand_pos[i]
      if obj_pos is not None:
        for i in range(obj_pos.shape[0]):
          self.sim.model.site_pos[self.model.site_name2id('obj_'+str(i+1))] = obj_pos[i]

      # assert hand_pos.shape[0] == obj_pos.shape[0]
      # num_states = hand_pos.shape[0]
      # for i in range(num_states):
      #     self.sim.model.site_pos[self.model.site_name2id('hand_'+str(i+1))] = hand_pos[i]
      #     self.sim.model.site_pos[self.model.site_name2id('obj_'+str(i+1))] = obj_pos[i]