#!/usr/bin/env python

import rospy
import actionlib
import tf
import moveit_commander
import pickle
import copy

import numpy as np
import IPython

from geometry_msgs.msg import *

from control_msgs.msg import FollowJointTrajectoryAction, FollowJointTrajectoryGoal
from trajectory_msgs.msg import JointTrajectory, JointTrajectoryPoint
from control_msgs.msg import PointHeadAction, PointHeadGoal
from control_msgs.msg import GripperCommandAction, GripperCommandGoal

from moveit_python import MoveGroupInterface, PlanningSceneInterface

from moveit_msgs.msg import *
from openravepy import * 
from prpy.rave import save_trajectory, load_trajectory
from sensor_util import SensorData
from sensor_msgs.msg import JointState

grasps = [
          np.array([-0.58229318,  0.19715234, -0.20228353, -0.68864361, -2.96338922,
       -1.98234405, -0.4500603]),  # object-0 (right to left - paper)
          np.array([-0.82504561,  0.19139977, -1.09199247, -1.21403203, -1.92296678,
       -1.83814944,  0.3015903]),  # object-1
          np.array([-0.61527374,  0.06944819, -1.43828854, -1.48171172, -1.64646679,
       -1.73115386,  1.33127493]),  # object-2 (clamp red)
          np.array([-0.37750676,  0.2404873 , -1.26303133, -1.55572626, -1.86889392,
       -1.88225116,  0.71384764]),
          np.array([0.2679157 ,  0.35055037, -0.47993407, -0.83475527, -2.5622533 ,
       -2.06786288,  0.71960009]),
          np.array([0.4677167 ,  0.23550196, -0.48070106, -0.55557075, -2.58718044,
       -1.84160079,  0.72036708])
        ]

########################## scenario 1 ################################################
places = [np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04]),
          np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04]),
          np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04]),
          np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04]),
          np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04]),
          np.array([1.26807121e+00,  2.07506855e-01, -5.65058002e-04, -6.36104794e-01,
       -1.20538578e-01,  1.04189931e+00, -2.20425298e-04])]

yay = np.array([0.8025, 0.960, 0.299, -2.245, -0.024, -0.834, 1.537])

grasp_aprh_dists = [0.12, 0.15, 0.095, 0.09, 0.08, 0.08]
grasp_lift_dists = [0.2, 0.2, 0.2, 0.2, 0.1, 0.1]
place_aprh_dists = [0.14, 0.14, 0.13, 0.13, 0.13, 0.13]

class FollowTrajectoryClient(object):

    def __init__(self, name, joint_names):
        self.client = actionlib.SimpleActionClient("%s/follow_joint_trajectory" % name,
                                                   FollowJointTrajectoryAction)
        rospy.loginfo("Waiting for %s..." % name)
        self.client.wait_for_server()
        self.joint_names = joint_names

    def move_to(self, positions, duration=2.0):
        if len(self.joint_names) != len(positions):
            rospy.loginfo("Invalid trajectory position!!!!!")
            return False
        trajectory = JointTrajectory()
        trajectory.joint_names = self.joint_names
        trajectory.points.append(JointTrajectoryPoint())
        trajectory.points[0].positions = positions
        trajectory.points[0].velocities = [0.0 for _ in positions]
        trajectory.points[0].accelerations = [0.0 for _ in positions]
        trajectory.points[0].time_from_start = rospy.Duration(duration)
        follow_goal = FollowJointTrajectoryGoal()
        follow_goal.trajectory = trajectory

        self.client.send_goal(follow_goal)
        self.client.wait_for_result()

class GripperCommandClient(object):

    def __init__(self):
        self.client = actionlib.SimpleActionClient("gripper_controller/gripper_action", GripperCommandAction)
        rospy.loginfo("Waiting for gripper_controller...")
        self.client.wait_for_server()

    def open(self):
        move = GripperCommandGoal()
        move.command.position = 0.10
        move.command.max_effort = 0.0
        rospy.loginfo("Command gripper to open ...")
        self.client.send_goal(move)
        self.client.wait_for_result()
        if (self.client.get_result().reached_goal == True):
            rospy.loginfo("Successfully placed object ...")
        else:
            rospy.loginfo("Failed placed object ...")
        
    def close(self):
        """
        Close the gripper and determine if an object was grasped.
        
        Returns:
            bool: True if an object was grasped, False otherwise
        """
        # Create the gripper command
        squeeze = GripperCommandGoal()
        squeeze.command.position = 0.0  # Fully closed position
        squeeze.command.max_effort = 80.0  # Maximum effort
        
        rospy.loginfo("Commanding gripper to close...")
        self.client.send_goal(squeeze)
        self.client.wait_for_result()
        
        # Get the result
        result = self.client.get_result()
        
        # Get the current gripper position
        gripper_position = result.position
        
        # Check if the gripper reached its goal
        if result.reached_goal:
            rospy.loginfo("Gripper closed completely (position: %.3f) - no object grasped", gripper_position)
            return False
        else:
            # The gripper didn't reach its goal, which means something is in the gripper
            rospy.loginfo("Gripper stopped at position %.3f - object successfully grasped", gripper_position)
            
            # You can add additional checks here if needed
            # For example, if the gripper barely moved, it might be a false positive
            if gripper_position > 0.08:  # If gripper is still mostly open
                rospy.logwarn("Gripper is still quite open - grasp may not be secure")
            
            return True

class PointHeadClient(object):

    def __init__(self):
        self.client = actionlib.SimpleActionClient("head_controller/point_head", PointHeadAction)
        rospy.loginfo("Waiting for head_controller...")
        self.client.wait_for_server()

    def look_at(self, x, y, z, frame, duration = 2.0):
        goal = PointHeadGoal()
        goal.target.header.stamp = rospy.Time.now()
        goal.target.header.frame_id = frame
        goal.target.point.x = x
        goal.target.point.y = y
        goal.target.point.z = z
        goal.min_duration = rospy.Duration(duration)
        self.client.send_goal(goal)
        self.client.wait_for_result()

class GraspingClient(GripperCommandClient):

    def __init__(self, env, robot, velocity_scale_factor, planner_id, n_objs): # removed env, robot, num_objs
        # initialize GripperCommandClient
        GripperCommandClient.__init__(self)
        
        # initialize Moveit MoveGroup and PlanningSceneInterface
        self.move_group = moveit_commander.MoveGroupCommander("arm")
        self.move_group.set_planner_id(planner_id)
        self.move_group.set_max_velocity_scaling_factor(velocity_scale_factor)
        self.move_group.set_num_planning_attempts(100)
        # Get joint names from the MoveGroupCommander
        self.joint_names = self.move_group.get_active_joints()

        self.planning_scene = PlanningSceneInterface("base_link")
        self.planning_scene.removeCollisionObject("my_table1")
        self.planning_scene.removeCollisionObject("my_ground")
        self.planning_scene.addBox("my_table1", 0.615, 1.25, 0.06, 0.764, -0.15, 0.65, wait=True)
        self.planning_scene.setColor("my_table1", 0.5, 0.5, 0.5)
        self.planning_scene.addBox("my_table2", 0.064, 1.25, 0.62, 0.651, -0.15, 0.31, wait=True)
        self.planning_scene.setColor("my_table2", 0.5, 0.5, 0.5)
        self.planning_scene.addBox("my_ground", 8, 8, 0.2, 0.0, 0.0, -0.1, wait=True)
        self.planning_scene.setColor("my_ground", 0.37, 0.28, 0.18)
        self.planning_scene.sendColors()

        # create a torso client
        self.torso_client = FollowTrajectoryClient("torso_controller", ["torso_lift_joint"])
        # Initialize current joint positions
        self.current_joint_positions = [0.0] * len(self.joint_names)

        # initialize the robot in openrave simulator
        self.env = env
        self.robot = robot
        self.robot.SetActiveManipulator('arm')
        self.ikmodel = databases.inversekinematics.InverseKinematicsModel(self.robot,iktype=IkParameterization.Type.Transform6D)
        if not self.ikmodel.load():
            self.ikmodel.autogenerate()
        self.robot.SetActiveDOFs(self.ikmodel.manip.GetArmIndices())
        
        # scale robot's joints' v and a
        self.robot.SetDOFVelocityLimits(self.robot.GetDOFVelocityLimits()*velocity_scale_factor)
        self.robot.SetDOFAccelerationLimits(self.robot.GetDOFAccelerationLimits()*velocity_scale_factor)
        
        # setting up openrave motion planner
        self.or_planner_id ='OMPL_RRTConnect'
        self.planner = RaveCreatePlanner(self.env, self.or_planner_id)
        self.simplifier = RaveCreatePlanner(self.env, 'OMPL_Simplifier')
        self.params = Planner.PlannerParameters()
        self.params.SetMaxIterations(1000)
        self.params.SetRobotActiveJoints(self.robot)
        self.params.SetExtraParameters('<range>0.02</range>')

        # create a tf listener
        self.listener = tf.TransformListener()

        # load trajs for grasping and placing
        self.scenario = '4'
        self.grasp_trajs = []
        self.place_trajs = []
        self.pass_grasp_trajs = []
        self.pass_place_trajs = []
        self.ready_l_trajs = []
        self.ready_p_trajs = []
        self.found_objs = []
        self.replan = rospy.get_param('/fetch_robot_client/replan')
        self.n_objs = n_objs
        if not self.replan:
            for i in range(n_objs):
                self.grasp_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/grasp_trajs/' + self.scenario + '_grasp_traj_' + str(i) + '.xml'))
                self.place_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/place_trajs/' + self.scenario + '_place_traj_' + str(i) + '.xml'))
                self.pass_grasp_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/grasp_trajs/' + self.scenario + '_pass_grasp_traj_' + str(i) + '.xml'))
                self.pass_place_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/place_trajs/' + self.scenario + '_pass_place_traj_' + str(i) + '.xml'))
                self.ready_l_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/ready_l_trajs/' + self.scenario + '_ready_l_traj_' + str(i) + '.xml'))
                self.ready_p_trajs.append(load_trajectory(self.env, '/home/fetch/projects/src/warehouse_domain/scripts/utils/ready_p_trajs/' + self.scenario + '_ready_p_traj_' + str(i) + '.xml'))

        self.head_client = PointHeadClient()
        self.head_camera_client = SensorData(self.env)

        self.manip_objs = [0,0,0]
        
    def joint_state_callback(self, msg):
        # Update current joint positions from the joint states message
        joint_indices = {name: index for index, name in enumerate(msg.name)}
        for i, joint_name in enumerate(self.joint_names):
            if joint_name in joint_indices:
                self.current_joint_positions[i] = msg.position[joint_indices[joint_name]]

    def set_wrist_velocity(self, roll_velocity, flex_velocity, duration=1.0):
        """
        Set velocity for wrist_roll_joint and wrist_flex_joint for a specified duration,
        ensuring the joints don't exceed their limits.
        
        Args:
            roll_velocity: Velocity for wrist_roll_joint in rad/s
            flex_velocity: Velocity for wrist_flex_joint in rad/s
            duration: How long to run at this velocity (in seconds)
            
        Returns:
            bool: Success or failure
        """
        try:
            # Import necessary message types
            from sensor_msgs.msg import JointState
            
            # Create action client for arm controller
            arm_client = actionlib.SimpleActionClient(
                "arm_controller/follow_joint_trajectory", 
                FollowJointTrajectoryAction)
            
            # Wait for the action server to start
            rospy.loginfo("Waiting for arm_controller action server...")
            if not arm_client.wait_for_server(rospy.Duration(5.0)):
                rospy.logerr("Timed out waiting for arm controller action server")
                return False
            
            # Get current joint states
            try:
                joint_state_msg = rospy.wait_for_message("/joint_states", JointState, timeout=2.0)
            except rospy.ROSException as e:
                rospy.logerr("Failed to get joint states: {}".format(e))
                return False
            
            # Define the joint names for the arm
            arm_joint_names = [
                "shoulder_pan_joint", 
                "shoulder_lift_joint", 
                "upperarm_roll_joint", 
                "elbow_flex_joint", 
                "forearm_roll_joint", 
                "wrist_flex_joint", 
                "wrist_roll_joint"
            ]
            
            # Extract current positions for arm joints
            current_positions = []
            for joint_name in arm_joint_names:
                if joint_name in joint_state_msg.name:
                    index = joint_state_msg.name.index(joint_name)
                    current_positions.append(joint_state_msg.position[index])
                else:
                    rospy.logerr("Joint {} not found in joint states".format(joint_name))
                    return False
            
            # Get joint limits from MoveIt
            wrist_flex_index = arm_joint_names.index("wrist_flex_joint")
            wrist_roll_index = arm_joint_names.index("wrist_roll_joint")
            
            # Fetch joint limits
            # These are the standard limits for Fetch robot - adjust if your robot has different limits
            joint_limits = {
                "wrist_flex_joint": {"min": -1.5, "max": 2},  # Typical Fetch limits
                "wrist_roll_joint": {"min": -2.5, "max": 2.5}   # Typical Fetch limits
            }
            
            # Check if current positions are already at or near limits and adjust velocities accordingly
            adjusted_flex_velocity = flex_velocity
            adjusted_roll_velocity = roll_velocity
            
            # Check wrist_flex_joint
            if flex_velocity > 0 and current_positions[wrist_flex_index] >= joint_limits["wrist_flex_joint"]["max"] - 0.01:
                rospy.logwarn("Wrist flex already at maximum limit. Setting flex velocity to 0.")
                adjusted_flex_velocity = 0.0
            elif flex_velocity < 0 and current_positions[wrist_flex_index] <= joint_limits["wrist_flex_joint"]["min"] + 0.01:
                rospy.logwarn("Wrist flex already at minimum limit. Setting flex velocity to 0.")
                adjusted_flex_velocity = 0.0
            
            # Check wrist_roll_joint
            if roll_velocity > 0 and current_positions[wrist_roll_index] >= joint_limits["wrist_roll_joint"]["max"] - 0.01:
                rospy.logwarn("Wrist roll already at maximum limit. Setting roll velocity to 0.")
                adjusted_roll_velocity = 0.0
            elif roll_velocity < 0 and current_positions[wrist_roll_index] <= joint_limits["wrist_roll_joint"]["min"] + 0.01:
                rospy.logwarn("Wrist roll already at minimum limit. Setting roll velocity to 0.")
                adjusted_roll_velocity = 0.0
            
            # Calculate end positions based on velocity and duration
            end_positions = list(current_positions)
            
            # Calculate potential end position for wrist_flex_joint
            potential_flex_end = current_positions[wrist_flex_index] + adjusted_flex_velocity * duration
            
            # Check against limits and adjust if necessary
            if potential_flex_end > joint_limits["wrist_flex_joint"]["max"]:
                rospy.logwarn("Wrist flex would exceed maximum limit. Setting end position to limit.")
                end_positions[wrist_flex_index] = joint_limits["wrist_flex_joint"]["max"]
            elif potential_flex_end < joint_limits["wrist_flex_joint"]["min"]:
                rospy.logwarn("Wrist flex would exceed minimum limit. Setting end position to limit.")
                end_positions[wrist_flex_index] = joint_limits["wrist_flex_joint"]["min"]
            else:
                # Within limits, use calculated position
                end_positions[wrist_flex_index] = potential_flex_end
            
            # Calculate potential end position for wrist_roll_joint
            potential_roll_end = current_positions[wrist_roll_index] + adjusted_roll_velocity * duration
            
            # Check against limits and adjust if necessary
            if potential_roll_end > joint_limits["wrist_roll_joint"]["max"]:
                rospy.logwarn("Wrist roll would exceed maximum limit. Setting end position to limit.")
                end_positions[wrist_roll_index] = joint_limits["wrist_roll_joint"]["max"]
            elif potential_roll_end < joint_limits["wrist_roll_joint"]["min"]:
                rospy.logwarn("Wrist roll would exceed minimum limit. Setting end position to limit.")
                end_positions[wrist_roll_index] = joint_limits["wrist_roll_joint"]["min"]
            else:
                # Within limits, use calculated position
                end_positions[wrist_roll_index] = potential_roll_end
            
            # Create a trajectory with two points - start and end
            trajectory = JointTrajectory()
            trajectory.joint_names = arm_joint_names
            
            # Add current position as first point
            trajectory.points.append(JointTrajectoryPoint())
            trajectory.points[0].positions = current_positions
            
            # Set initial velocities
            initial_velocities = [0.0] * len(arm_joint_names)
            initial_velocities[wrist_flex_index] = adjusted_flex_velocity
            initial_velocities[wrist_roll_index] = adjusted_roll_velocity
            
            trajectory.points[0].velocities = initial_velocities
            trajectory.points[0].accelerations = [0.0] * len(arm_joint_names)
            trajectory.points[0].time_from_start = rospy.Duration(0.0)
            
            # Add end position as second point
            trajectory.points.append(JointTrajectoryPoint())
            trajectory.points[1].positions = end_positions
            
            # Set velocities for end point (should be zero to stop)
            end_velocities = [0.0] * len(arm_joint_names)
            trajectory.points[1].velocities = end_velocities
            trajectory.points[1].accelerations = [0.0] * len(arm_joint_names)
            trajectory.points[1].time_from_start = rospy.Duration(duration)
            
            # Create and send goal
            goal = FollowJointTrajectoryGoal()
            goal.trajectory = trajectory
            
            rospy.loginfo("Sending wrist velocity command: roll={} rad/s, flex={} rad/s for {} seconds".format(
                adjusted_roll_velocity, adjusted_flex_velocity, duration))
            arm_client.send_goal(goal)
            
            # Wait for result with timeout (duration + buffer)
            if arm_client.wait_for_result(rospy.Duration(duration + 2.0)):
                result = arm_client.get_result()
                rospy.loginfo("Wrist velocity command completed")
                return True
            else:
                rospy.logerr("Wrist velocity command timed out")
                arm_client.cancel_goal()
                return False
            
        except Exception as e:
            rospy.logerr("Error in set_wrist_velocity: {}".format(e))
            return False
        
    def ready_to_grasp(self):
        start = self.robot.GetActiveDOFValues()
        # if start[0] > 0:
        #     goal = np.array([1.32, 0.7, 0.0, -2.0, 0.0, -0.57, 0.0])
        # else:
        goal = np.array([-1.32, 0.7, 0.0, -2.0, 0.0, -0.57, 0.0])
        traj = self.or_MP(goal)
        rospy.loginfo("Ready to grasp")
        return traj
    
    def ready_to_hold(self):
        start = self.robot.GetActiveDOFValues()
        goal = np.array([0.0, 1.4, 0.0, -2.2, 0.0, 0.7, 1.55])
        traj = self.or_MP(goal)
        rospy.loginfo("Ready to grasp")
        return traj
    
    def ready_to_throw(self):
        start = self.robot.GetActiveDOFValues()
        # if start[0] > 0:
        #     goal = np.array([1.32, 0.7, 0.0, -2.0, 0.0, -0.57, 0.0])
        # else:
        # shoulder_joint, elbow_joint, wrist_joint, wrist_flex_joint, wrist_roll_joint, gripper_joint
        goal = np.array([-1.32, 0.6, -1.0, -2.0, 0.0, -0.57, 0.0])
        traj = self.or_MP(goal)
        rospy.loginfo("Ready to grasp")
        return traj

    def yay(self):
        goal = yay
        traj = self.or_MP(goal)
        return traj

    def look_for_obj(self, obj_id):
        n_obj_id = self.manip_objs[obj_id]*3 + obj_id
        self.manip_objs[obj_id] += 1

        rospy.sleep(1.0)
        # move to pre-grasp pose
        self.robot.ExecutePath(self.grasp_trajs[n_obj_id])
        rospy.sleep(0.5)
        
        # approach the obj
        self.approach(grasp_aprh_dists[n_obj_id])

        # close gripper
        self.close()

        # lift the obj
        self.lift(dist=grasp_lift_dists[n_obj_id])
        rospy.sleep(0.5)

        # place obj at ready spot
        if n_obj_id == 5:
            pose = self.get_gripper_pose()
            pose[0,3] = 0.55
            pose[1,3] = -0.5
            pose[2,3] -= 0.075

            with self.env:
                sol = self.ikmodel.manip.FindIKSolution(pose, IkFilterOptions.CheckEnvCollisions)
                if sol is None:
                    #sol = np.array([0.778, -0.477, -0.125, 0.869, 0.104, 1.158, 1.214])
                    IPython.embed()
            traj = self.or_MP(sol)
        else:
            self.robot.ExecutePath(self.place_trajs[n_obj_id])
        rospy.sleep(0.5)
        
        # release the obj
        self.open()

        # move away gripper
        if n_obj_id == 5:
            self.lift(dist=0.115)
        else:
            self.lift(dist=0.12)
        rospy.sleep(1.0)
        
        if n_obj_id == 5:
            self.ready_to_grasp()
        else:
            # move back to default pose
            self.robot.ExecutePath(self.ready_l_trajs[n_obj_id])
        rospy.sleep(0.5)

        self.found_objs.append(n_obj_id)

    def pass_obj(self, TB_id):

        if TB_id == 1:
            obj_id = self.found_objs.pop()
        else:
            obj_id = self.found_objs.pop(0)

        # execute pass action
        self.robot.ExecutePath(self.pass_grasp_trajs[obj_id])
        rospy.sleep(0.5)
        
        # approach the obj
        self.approach(place_aprh_dists[obj_id])

        rospy.sleep(0.5)
        # close gripper
        self.close()
        rospy.sleep(0.5)
        
        # lift obj
        self.lift(dist=0.12)
        rospy.sleep(0.5)

        self.robot.ExecutePath(self.pass_place_trajs[obj_id])
        rospy.sleep(0.5)
        
        # release obj
        self.open()
        rospy.sleep(1.0)

        # move back to default arm pose
        #self.robot.ExecutePath(self.ready_p_trajs[obj_id])
        self.ready_to_grasp()

    def move_arm(self, goal_id): #pick and place demo

        rospy.loginfo('Start for recording trajectories ...')
        self.open()
        rospy.sleep(0.2)
        # import pdb
        # pdb.set_trace()

        # move to pre-grasp pose
        if self.replan:
            traj = self.or_MP(grasps[goal_id])
            self.save_grasp_traj(traj, goal_id)
        else:
            self.robot.ExecutePath(self.grasp_trajs[goal_id])
            
        self.open()
        rospy.sleep(0.5)

        # approach the obj
        self.approach(grasp_aprh_dists[goal_id])
        
        
        # Lower vertically to grasp
        self.lower_with_openrave(0.1)  # Lower by 8cm
    

        # close gripper
        self.close()

        # lift obj
        self.lift(dist=grasp_lift_dists[goal_id])
        rospy.sleep(0.5)

        # pass obj
        if self.replan:
            traj = self.or_MP(places[goal_id])
            self.save_pass_place_traj(traj, goal_id)
        else:
            self.robot.ExecutePath(self.pass_place_trajs[goal_id])

        # release obj
        self.open()
        #self.lift(dist=0.12)
        rospy.sleep(0.5)

        # move to ready pose
        if self.replan:
            traj = self.ready_to_grasp()
            self.save_ready_p_traj(traj, goal_id)
        else:
            self.robot.ExecutePath(self.ready_p_trajs[goal_id])
    def lower_with_openrave(self, distance=0.1, max_attempts=5):
        """
        Lower the gripper using OpenRAVE planning with replanning capabilities.
        
        Args:
            distance: Distance to lower in meters
            max_attempts: Maximum number of planning attempts
            
        Returns:
            bool: Success or failure
        """
        rospy.loginfo("Attempting to lower gripper by %.3f meters using OpenRAVE", distance)
        
        # Ensure ROS spinner is running for MoveIt communication
        try:
            # This is a Python wrapper to start the AsyncSpinner
            # Note: In Python, we use rospy.spin() or a MultiThreadedSpinner
            # We'll use a separate thread to handle callbacks
            import threading
            def spin_thread():
                rospy.spin()
            
            spinner_thread = threading.Thread(target=spin_thread)
            spinner_thread.daemon = True
            spinner_thread.start()
            rospy.loginfo("Started ROS spinner thread for MoveIt communication")
        except Exception as e:
            rospy.logwarn("Failed to start spinner thread: %s", str(e))
        
        # Get current gripper pose
        current_pose = self.get_gripper_pose()
        
        # Create target pose (lower)
        target_pose = np.copy(current_pose)
        target_pose[2, 3] -= distance  # Subtract from Z to move down
        
        # Get current joint values
        current_joints = self.robot.GetActiveDOFValues()
        
        # Find IK solution for target pose (ignoring collisions with target object)
        with self.env:
            target_sol = self.ikmodel.manip.FindIKSolution(
                target_pose, IkFilterOptions.IgnoreEndEffectorCollisions)
            
            if target_sol is None:
                rospy.logwarn("No IK solution exists for target pose")
                return False
        
        # Try different planning strategies with OpenRAVE
        for attempt in range(max_attempts):
            rospy.loginfo("OpenRAVE planning attempt %d/%d", attempt+1, max_attempts)
            
            try:
                # Set up the planner parameters
                params = Planner.PlannerParameters()
                params.SetRobotActiveJoints(self.robot)
                params.SetGoalConfig(target_sol)
                params.SetExtraParameters('<range>0.01</range>')  # Smaller step size
                
                if attempt > 0:
                    # Increase iterations for later attempts
                    params.SetMaxIterations(1000 * (attempt + 1))
                
                # Select planner based on attempt
                if attempt == 0:
                    planner_name = self.or_planner_id
                elif attempt == 1:
                    planner_name = "BiRRT"
                elif attempt == 2:
                    planner_name = "RRTConnect"
                else:
                    planner_name = "BasicRRT"
                
                # Create and initialize the planner
                planner = RaveCreatePlanner(self.env, planner_name)
                if not planner.InitPlan(self.robot, params):
                    rospy.logwarn("Failed to initialize OpenRAVE planner %s", planner_name)
                    continue
                
                # Plan trajectory
                traj = RaveCreateTrajectory(self.env, '')
                result = planner.PlanPath(traj)
                
                if result == PlannerStatus.HasSolution and traj.GetNumWaypoints() > 0:
                    rospy.loginfo("OpenRAVE planning successful with %s", planner_name)
                    
                    # Execute the trajectory with OpenRAVE
                    try:
                        rospy.loginfo("Executing OpenRAVE trajectory")
                        self.robot.ExecutePath(traj)
                        rospy.loginfo("Successfully lowered gripper using OpenRAVE")
                        return True
                    except Exception as e:
                        rospy.logwarn("OpenRAVE execution failed: %s", str(e))
                        
                        # If OpenRAVE execution fails, try executing with MoveIt
                        try:
                            rospy.loginfo("Trying to execute with MoveIt")
                            
                            # Extract waypoints from OpenRAVE trajectory
                            waypoints = []
                            for i in range(traj.GetNumWaypoints()):
                                waypoint = traj.GetWaypoint(i)
                                dof_values = traj.GetConfigurationSpecification().ExtractJointValues(
                                    waypoint, self.robot, self.robot.GetActiveDOFIndices())
                                waypoints.append(dof_values)
                            
                            # Create a RobotTrajectory for MoveIt
                            robot_trajectory = RobotTrajectory()
                            robot_trajectory.joint_trajectory.joint_names = self.move_group.get_active_joints()
                            
                            # Add waypoints to trajectory
                            time_from_start = 0.0
                            for i, wp in enumerate(waypoints):
                                point = JointTrajectoryPoint()
                                point.positions = wp
                                point.velocities = [0.0] * len(wp)
                                
                                # Increase time between waypoints
                                time_from_start += 0.5  # 0.5 seconds between waypoints
                                point.time_from_start = rospy.Duration(time_from_start)
                                
                                robot_trajectory.joint_trajectory.points.append(point)
                            
                            # Execute the trajectory with MoveIt
                            self.move_group.execute(robot_trajectory, wait=True)
                            rospy.loginfo("Successfully executed OpenRAVE trajectory with MoveIt")
                            return True
                        except Exception as e:
                            rospy.logwarn("MoveIt execution failed: %s", str(e))
                            
                            # If MoveIt fails, try with direct action client
                            try:
                                rospy.loginfo("Trying to execute with direct action client")
                                
                                # Create a new trajectory for the action client
                                joint_traj = JointTrajectory()
                                joint_traj.joint_names = self.move_group.get_active_joints()
                                
                                # Add waypoints to trajectory
                                time_from_start = 0.0
                                for i, wp in enumerate(waypoints):
                                    point = JointTrajectoryPoint()
                                    point.positions = wp
                                    point.velocities = [0.0] * len(wp)
                                    
                                    # Increase time between waypoints
                                    time_from_start += 0.5  # 0.5 seconds between waypoints
                                    point.time_from_start = rospy.Duration(time_from_start)
                                    
                                    joint_traj.points.append(point)
                                
                                # Create and send goal
                                goal = FollowJointTrajectoryGoal()
                                goal.trajectory = joint_traj
                                
                                # Send the goal to the action server
                                client = actionlib.SimpleActionClient(
                                    "arm_controller/follow_joint_trajectory",
                                    FollowJointTrajectoryAction)
                                client.wait_for_server()
                                client.send_goal(goal)
                                
                                # Wait for result with timeout
                                if client.wait_for_result(rospy.Duration(20.0)):
                                    result = client.get_result()
                                    rospy.loginfo("Successfully executed OpenRAVE trajectory with action client")
                                    return True
                                else:
                                    rospy.logwarn("Action client execution timed out")
                                    client.cancel_goal()
                            except Exception as e:
                                rospy.logwarn("Action client execution failed: %s", str(e))
                else:
                    rospy.logwarn("OpenRAVE planning failed with %s", planner_name)
            
            except Exception as e:
                rospy.logwarn("Error during OpenRAVE planning attempt %d: %s", attempt+1, str(e))
        
        # If all OpenRAVE planning attempts fail, try a direct approach with MoveIt
        rospy.loginfo("All OpenRAVE planning attempts failed, trying direct joint movement with MoveIt")
        
        try:
            # Set joint target directly with MoveIt
            self.move_group.set_max_velocity_scaling_factor(0.1)  # Slow down
            self.move_group.set_joint_value_target(target_sol)
            success = self.move_group.go(wait=True)
            self.move_group.set_max_velocity_scaling_factor(0.5)  # Reset speed
            
            if success:
                rospy.loginfo("Successfully lowered gripper using MoveIt direct joint movement")
                return True
            else:
                rospy.logwarn("MoveIt direct joint movement failed")
        except Exception as e:
            rospy.logwarn("Error during MoveIt direct joint movement: %s", str(e))
        
        # Final fallback: try with direct action client
        try:
            rospy.loginfo("Attempting final fallback: direct action client")
            
            # Create a simple trajectory to move directly to target
            joint_traj = JointTrajectory()
            joint_traj.joint_names = self.move_group.get_active_joints()
            
            # Add current position as first point
            joint_traj.points.append(JointTrajectoryPoint())
            joint_traj.points[0].positions = current_joints
            joint_traj.points[0].velocities = [0.0] * len(current_joints)
            joint_traj.points[0].time_from_start = rospy.Duration(0.0)
            
            # Add target position as second point with longer duration
            joint_traj.points.append(JointTrajectoryPoint())
            joint_traj.points[1].positions = target_sol
            joint_traj.points[1].velocities = [0.0] * len(current_joints)
            joint_traj.points[1].time_from_start = rospy.Duration(3.0)  # 3 seconds for movement
            
            # Create and send goal
            goal = FollowJointTrajectoryGoal()
            goal.trajectory = joint_traj
            
            # Send the goal to the action server
            client = actionlib.SimpleActionClient(
                "arm_controller/follow_joint_trajectory",
                FollowJointTrajectoryAction)
            client.wait_for_server()
            client.send_goal(goal)
            
            # Wait for result with timeout
            if client.wait_for_result(rospy.Duration(10.0)):
                result = client.get_result()
                rospy.loginfo("Successfully lowered gripper using direct action client")
                return True
            else:
                rospy.logwarn("Direct action client timed out")
                client.cancel_goal()
        except Exception as e:
            rospy.logwarn("Error during direct action client: %s", str(e))
        
        rospy.logerr("Failed to lower gripper after all attempts")
        return False

    def approach(self, dist=0.08):
        way_points = self.create_waypoints_approach(dist)
        (plan, fraction) = self.move_group.compute_cartesian_path(way_points, 0.001, 0.0, False)
        if fraction == 1.0:
            self.move_group.execute(plan)

    def lift(self, dist=0.08):
        way_points = self.create_waypoints_lift(dist)
        (plan, fraction) = self.move_group.compute_cartesian_path(way_points, 0.001, 0.0, False)
        if fraction == 1.0:
            self.move_group.execute(plan)
 
    ############################# Helper Functions #########################################
    
    def or_MP(self, goal, placing=False):
        self.params.SetRobotActiveJoints(self.robot)
        self.params.SetGoalConfig(goal)
        
        with self.env:
            with self.robot:
                rospy.loginfo('Calling the planner ...')
                traj = RaveCreateTrajectory(self.env, '')
                self.planner.InitPlan(self.robot, self.params)
                result = self.planner.PlanPath(traj)
                assert result == PlannerStatus.HasSolution
        
        rospy.loginfo('Waiting for the real robot finishing the movement ...')
        self.robot.ExecutePath(traj)
        return traj

    def create_waypoints_approach(self, dist):
        pose = self.get_gripper_pose()
        waypoints = []
        waypoints.append(self.move_group.get_current_pose().pose)
        point = copy.deepcopy(waypoints[0])
        point.position.x += dist * pose[0,0]
        point.position.y += dist * pose[1,0]
        point.position.z += dist * pose[2,0]
        waypoints.append(point)
        return waypoints

    def create_waypoints_lift(self, dist):
        waypoints = []
        waypoints.append(self.move_group.get_current_pose().pose)
        point = copy.deepcopy(waypoints[0])
        point.position.z += dist
        waypoints.append(point)
        return waypoints

    def get_gripper_pose(self):
        count = 0
        while count < 20:
            (trans, quat) = self.listener.lookupTransform('base_link', 'wrist_roll_link', rospy.Time(0))
            count += 1
        pose = tf.transformations.concatenate_matrices(tf.transformations.translation_matrix(trans),tf.transformations.quaternion_matrix(quat))
        return pose

    def save_grasp_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/grasp_trajs/' + self.scenario + '_grasp_traj_' + str(id) + '.xml')
            
    def save_place_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/place_trajs/' + self.scenario + '_place_traj_' + str(id) + '.xml')
        
    def save_pass_grasp_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/grasp_trajs/' + self.scenario + '_pass_grasp_traj_' + str(id) + '.xml')
            
    def save_pass_place_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/place_trajs/' + self.scenario + '_pass_place_traj_' + str(id) + '.xml')

    def save_ready_l_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/ready_l_trajs/' + self.scenario + '_ready_l_traj_' + str(id) + '.xml')

    def save_ready_p_traj(self, traj, id):
        save_trajectory(traj, '/home/fetch/projects/src/warehouse_domain/scripts/utils/ready_p_trajs/' + self.scenario + '_ready_p_traj_' + str(id) + '.xml')
