#!/usr/bin/env python

import numpy as np
import rospy
import fetchpy
import tf
import time
import pdb
import logging
import random

from openravepy import *
from warehouse_domain.srv import *
from warehouse_domain.msg import CenCtrlResp, BasketObjs, ReadyObjs, Loc, ObsX

from utils.manip_util import *
from utils.sensor_util import SensorData

# openrave env
scenario_path = "/home/fetch/projects/src/warehouse_domain/or_envs/scenario0_real.xml"
# moveit planner
moveit_planner_id = 'ESTkConfigDefault'

# collect global variables
num_steps = rospy.get_param('/run_fetch_balance/num_steps')


class FetchRobot(object):

    def __init__(self):
        self.idx = 2
        self.run_real_fetch = rospy.get_param("/run_fetch_balance/run_real_fetch")


        # create a tf listener to get turtlebot's position
        if self.run_real_fetch:
            self.pos_listener = tf.TransformListener()

        # indicate the init step
        self.init = True


        rospy.loginfo("Finish initialization ...")

        ############################# for real fetch robol controller #########################

        # setup torso, head and arm controllers
        if self.run_real_fetch:
            self.torso_client = FollowTrajectoryClient("torso_controller", ["torso_lift_joint"])
            self.head_client = PointHeadClient()

            # # raising the torso
            # rospy.loginfo("Raising torso ...")
            # self.torso_client.move_to([0.4])
            # rospy.sleep(1)

            # # moving head
            # rospy.loginfo("Looking at ...")
            # self.head_client.look_at(1.6, -0.0, 0.0, "base_link")
            # rospy.sleep(1)

            # setting up environment in OpenRave simulator
            fetchpy_args = {'sim':False,'viewer':'qtcoin','robot_xml':None,'env_path':scenario_path,'base_sim':False,'perception_sim': False}
            self.env, self.robot = fetchpy.initialize(**fetchpy_args)

            self.manip_client = GraspingClient(self.env, self.robot, 0.52, moveit_planner_id, 1)
            
            # self.manip_client.open()
            # self.manip_client.ready_to_hold()
            # rospy.sleep(1) #use this time to place the plate into the gripper
            # self.manip_client.close()
            
            # setup sensor client
            self.head_camera_client = SensorData(self.env)
            # self.head_camera_client.UpdateEnvPointCloud(name="cloud0")

            self.force_wait = False
            
            # rospy.loginfo("Lowering torso to position...")
            # self.torso_client.move_to([0.25])
            # rospy.sleep(1)

            # rospy.loginfo("Before breakpoint")
            # pdb.set_trace()
            # rospy.loginfo("After breakpoint")

            # rospy.loginfo("3")
            # rospy.sleep(1)
            # rospy.loginfo("2")
            # rospy.sleep(1)
            # rospy.loginfo("1")
            # rospy.sleep(1)
            
    def sample_random_policy(self):
        """
        Sample a random policy for wrist velocities.
        
        Returns:
            tuple: A tuple containing the sampled wrist velocities (roll_velocity, flex_velocity).
        """
        # Randomly choose to either sample a velocity or return (0, 0)
        if random.random() < 0.5:  # 50% chance to return (0, 0)
            return (0.0, 0.0)
        else:
            # Sample wrist velocities between -1 and 1
            roll_velocity = random.uniform(-1.0, 1.0)
            flex_velocity = random.uniform(-1.0, 1.0)
            return (roll_velocity, flex_velocity)

    def run(self):
        """A while loop in this function let the robot keep calling the controller service
           and executing the returned macro-action until finishing the entire task.
        """
        while not rospy.is_shutdown():
            
            # observation = self.head_camera_client.GetDepth()
            # action = self.policy(observation)
            # self.execute(action)
            action = self.sample_random_policy()
            self.execute(action)
            
            # # obtain macro observation
            # macro_obs = self._get_macro_obs()
            # #rospy.loginfo("Get new macro-obs ...")

            # # call cen_controller service to request macro-action
            # # rospy.loginfo("Call centralized controller ...")
            # if self.num_passed_obj < NumEachObj * NumDiffObjs:
            #     rospy.loginfo("The TBs beside the table are %s"%(self.TBAtTable))
            # re = self.policy(ObsX(self.idx, macro_obs))
            # macro_a = re.ma.ma
            # # self.logger.info(ACTION_NAMES[macro_a])
            # #rospy.loginfo("Obtain macro-action ...")

            # if self.num_passed_obj < NumEachObj * NumDiffObjs:
            #     # execute macro-action
            #     self.execute(macro_a)
            # else:
            #     pass

    def execute(self, action):
        wrist_flex_velocity = action[0]
        wrist_roll_velocity = action[1]
        self.manip_client.set_wrist_velocity(wrist_flex_velocity, wrist_roll_velocity,1)

    ################################ Helper Functions #######################################
    def _locate_ball(self):
        pass

    def setup_logger(self, name, log_file, level=logging.INFO):
        handler = logging.FileHandler(log_file)
        handler.setFormatter(self.formatter)

        logger = logging.getLogger(name)
        logger.setLevel(level)
        logger.addHandler(handler)
        return logger

if __name__ == "__main__":
    rospy.init_node('run_fetch_balance')
    agent = FetchRobot()
    agent.run()
