Skip to content

Commit

Permalink
Port interbotix_xsarm_dual package to ROS 2 (#170)
Browse files Browse the repository at this point in the history
* [xsarm_dual] Port xsarm_dual demo to ROS 2

* Fix DeclareLaunchArgument ordering

* Fix some DeclareLaunchArgument descriptions

* Fix description declaration kwargs, silence RViz

* Port over xsarm_dual script

* Fix robot names, delay type, add descriptive comments

* Clarify comment
  • Loading branch information
lukeschmitt-tr authored Feb 26, 2024
1 parent 777c271 commit 63562f0
Show file tree
Hide file tree
Showing 8 changed files with 493 additions and 253 deletions.
50 changes: 36 additions & 14 deletions interbotix_ros_xsarms/examples/interbotix_xsarm_dual/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -1,20 +1,42 @@
cmake_minimum_required(VERSION 3.0.2)
cmake_minimum_required(VERSION 3.5)
project(interbotix_xsarm_dual)

# Default to C99
if(NOT CMAKE_C_STANDARD)
set(CMAKE_C_STANDARD 99)
endif()

## Find catkin macros and libraries
## if COMPONENTS list like find_package(catkin REQUIRED COMPONENTS xyz)
## is used, also find other catkin packages
find_package(catkin REQUIRED COMPONENTS
interbotix_xsarm_control
interbotix_xs_modules
# Default to C++14
if(NOT CMAKE_CXX_STANDARD)
set(CMAKE_CXX_STANDARD 17)
endif()

if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
add_compile_options(-Wall -Wextra -Wpedantic)
endif()

find_package(ament_cmake REQUIRED)

# Install Python executables
install(
PROGRAMS
demos/xsarm_dual.py
DESTINATION
lib/${PROJECT_NAME}
)

###################################
## catkin specific configuration ##
###################################
## The catkin_package macro generates cmake config files for your package
## Declare things to be passed to dependent projects
catkin_package(
CATKIN_DEPENDS interbotix_xsarm_control interbotix_xs_modules
install(
DIRECTORY
config
launch
rviz
DESTINATION
share/${PROJECT_NAME}/
)

if(BUILD_TESTING)
find_package(ament_lint_auto REQUIRED)
ament_lint_auto_find_test_dependencies()
endif()

ament_package()
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,116 @@
#!/usr/bin/env python3

# Copyright 2024 Trossen Robotics
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
#
# * Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
#
# * Neither the name of the copyright holder nor the names of its
# contributors may be used to endorse or promote products derived from
# this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
# POSSIBILITY OF SUCH DAMAGE.


import math

from interbotix_xs_modules.xs_robot.arm import InterbotixManipulatorXS
from rclpy.duration import Duration

"""
This script is used to make two WidowX-200 arms work in tandem with one another
To get started, open a terminal and type:
ros2 launch interbotix_xsarm_dual xsarm_dual.launch.py
Then change to this directory and type:
python3 xsarm_dual.py
Note that the 'robot_name' argument used when instantiating an InterbotixManipulatorXS instance is
the same name as the 'robot_name_X' launch file argument.
To simultaneously control two arms, we don't wait for movements to finish by setting blocking to
False and delays to 0; opting instead for a function called after each movement command that uses
the node-owning robot's core clock's `sleep_for()` method. This allows us to send commands to both
robots at the same time and wait for their movements to complete after.
"""

MOVING_TIME_S = 2
SLEEP_DURATION = Duration(seconds=MOVING_TIME_S)


def main():
# robot_1 is the "node owner", meaning that it controls the state of rclpy.
robot_1 = InterbotixManipulatorXS(
robot_model='wx200',
robot_name='arm_1',
moving_time=MOVING_TIME_S,
node_owner=True,
)

# Because robot_1 is the node owner, we set this one's node_owner arg to False.
robot_2 = InterbotixManipulatorXS(
robot_model='wx200',
robot_name='arm_2',
moving_time=MOVING_TIME_S,
node_owner=False,
)

# Helper function used to wait for the robots' pre-configured moving times.
def wait() -> None:
"""Sleep for SLEEP_DURATION."""
robot_1.core.get_clock().sleep_for(SLEEP_DURATION)

robot_1.arm.go_to_home_pose(blocking=False)
robot_2.arm.go_to_home_pose(blocking=False)
wait()

robot_1.arm.set_ee_pose_components(x=0.3, z=0.2, blocking=False)
robot_2.arm.set_ee_pose_components(x=0.3, z=0.2, blocking=False)
wait()

robot_1.gripper.release(delay=0.0)
robot_2.gripper.release(delay=0.0)
wait()

robot_1.arm.set_single_joint_position('waist', -math.pi/4.0, blocking=False)
robot_2.arm.set_single_joint_position('waist', math.pi/4.0, blocking=False)
wait()

robot_1.gripper.grasp(delay=0.0)
robot_2.gripper.grasp(delay=0.0)
wait()

robot_1.arm.set_single_joint_position('waist', 0.0, blocking=False)
robot_2.arm.set_single_joint_position('waist', 0.0, blocking=False)
wait()

robot_1.arm.go_to_sleep_pose(blocking=False)
robot_2.arm.go_to_sleep_pose(blocking=False)
wait()

# Because robot_1 is the node owner, use it to do the shutdown process
robot_1.shutdown()


if __name__ == '__main__':
main()

This file was deleted.

Loading

0 comments on commit 63562f0

Please sign in to comment.