Skip to content

Latest commit

 

History

History
573 lines (442 loc) · 23.6 KB

README.md

File metadata and controls

573 lines (442 loc) · 23.6 KB

HEBI ROS 2 Examples

We provide three ways to control HEBI Arms using ROS 2:

  • Standalone HEBI API
  • ROS 2 Control
  • MoveIt

Standalone HEBI ROS 2 API

Set up your workspace using the following commands:

mkdir -p ~/hebi_ws/src
cd ~/hebi_ws/src
git clone -b ros2 https://github.com/HebiRobotics/hebi_cpp_api_ros.git
git clone -b ros2/$ROS_DISTRO https://github.com/HebiRobotics/hebi_description.git # ROS_DISTRO can be either humble, iron, or jazzy
git clone https://github.com/HebiRobotics/hebi_msgs.git
git clone https://github.com/HebiRobotics/hebi_ros2_examples.git

Once you have cloned the necessary repositories, install the required dependencies using rosdep

rosdep update
rosdep install --from-paths src --ignore-src -r -y

NOTE: If your ROS distribution is End-of-Life (EOL), you might need to include EOL distributions in your rosdep commands:

rosdep update --include-eol-distros
rosdep install --from-paths src --ignore-src -r -y --include-eol-distros

Then, build the workspace and source it using the following commands.

cd ~/hebi_ws
colcon build
source install/setup.bash

Create HRDF

To control your HEBI arm using the standalone API, you do not require a URDF file for robot description. Instead, we use a HRDF to describe the HEBI Arm.

For example, the HRDF for A-2085-05 arm is shown below.

<?xml version="1.0" encoding="UTF-8"?>
<!-- For documentation on the HEBI Robot Configuration please visit: -->
<!-- https://github.com/HebiRobotics/hebi-hrdf/blob/master/FORMAT.md -->

<!-- X-Series Arm (5-DoF) -->
<robot version="1.2.0">
  <actuator type="X8-9"/>
  <bracket type="X5HeavyRightOutside"/>
  <actuator type="X8-16"/>
  <link type="X5" extension="0.325" twist="pi"/>
  <actuator type="X8-9"/>
  <link type="X5" extension="0.325" twist="pi"/>
  <actuator type="X5-1"/>
  <bracket type="X5LightRight"/>
  <actuator type="X5-1"/>
  
  <!-- For custom end-effector parameters and formatting please visit: -->
  <!-- https://github.com/HebiRobotics/hebi-hrdf/blob/master/FORMAT.md#end-effector -->
  <end-effector />
  
</robot>

Arm Node

To control the arm with our arm_node, you need to create a parameter file <your_robot_name>_params.yaml file in config folder of the hebi_ros2_examples package.

The configuration file takes the following form:

arm_node:
  ros__parameters:
    names:
    families:
    gains_package:
    gains_file:
    hrdf_package:
    hrdf_file:
    home_position:

The parameters are defined as follows -

  • names: Array of "Names" of the HEBI Modules
  • families: Array of "Family" of HEBI Modules
  • gains_package: ROS package containing the gains file for your HEBI arm
  • gains_file: Relative path of the gains file from the gains_package
  • hrdf_package: ROS package containing the HRDF file for your HEBI arm
  • hrdf_file: Relative path of the HRDF file from the hrdf_package
  • home_position: Array of float values to move your arm after initialization.

NOTE:

  • names and families of your modules can be found and changed using Scope.

The configuration file for HEBI Arm A-2085-05 is given below.

arm_node:
  ros__parameters:
    names: [ "J1_base", "J2_shoulder", "J3_elbow", "J4_wrist1", "J5_wrist2" ]
    families: [ "HEBI" ]
    gains_package: "hebi_description"
    gains_file: "config/gains/A-2085-05_gains.xml"
    hrdf_package: "hebi_description"
    hrdf_file: "config/hrdf/A-2085-05.hrdf"
    home_position: [ 0.01, 2.09, 2.09, 1.5707963, 0.0 ]

Once you have both these files, you can run the arm node using the following command:

ros2 launch hebi_ros2_examples arm_node.launch.py hebi_arm:=<your_robot_name>

NOTE: Do not forget to build your workspace and source your setup before running the above command.

The arm node uses HEBI Arm API. It provides a variety of topics, services, and action for the user to control the arm.

Subscribers

  • /SE3_jog [control_msgs/msg/JointJog]: To command end effector jog in SE3 space (cartesian and rotation)
  • /cartesian_jog [control_msgs/msg/JointJog]: To command end effector jog in cartesian space (x, y, z)
  • /cartesian_trajectory [trajectory_msgs/msg/JointTrajectory]: To command a trajectory for the end effector in cartesian space
  • /joint_jog [control_msgs/msg/JointJog]: To command jog in joint angles
  • /joint_trajectory [trajectory_msgs/msg/JointTrajectory]: To command a trajectory in joint angles

Publishers

  • /ee_pose [geometry_msgs/msg/PoseStamped]: The end effector pose in SE3 space
  • /fdbk_joint_states [sensor_msgs/msg/JointState]: Joint angles of the arm
  • /inertia [geometry_msgs/msg/Inertia]: Inertia of the arm

Action Servers

  • /arm_motion [hebi_msgs/action/ArmMotion]:

Apart from the parameters set using the file discussed above, the arm node also uses few extra parameters which you can set dynamically:

  • compliant_mode: Setting it to true disables any goal set to the arm and sets the joint efforts to zero for easy manual movement of the arm.
  • ik_seed: This parameter sets the IK seed for inverse kinematic calculations.

To get you started, we have provided three example scripts that use the arm_node.

  1. ex_publish_trajectory: This published a predefined trajectory to the /joint_trajectory topic.
  2. ex_teach_repeat_mobileio: This uses mobile IO to record and play trajetories, or go to saved waypoints.
  3. ex_teleop_mobileio: This uses mobile IO to send jog commands to control the arm.

HRDF to URDF

NOTE: Still in development. the URDFs for the standard arm kits are already provided. Contact [email protected] if you need help with generating URDF

If you require ROS 2 control or MoveIt functionalities, you will need to have a URDF of the HEBI Arm. To help you with that, we have a script that converts HRDF to URDF, provided in hebi_description package.

The URDF for A-2085-05 looks like this

<?xml version='1.0' encoding='UTF-8'?>
<robot xmlns:xacro="http://www.ros.org/wiki/xacro" name="A-2085-05">
  
  <!-- HEBI A-2085-05 Arm Kit -->
  <xacro:include filename="$(find hebi_description)/urdf/components/hebi.xacro"/>

  <link name="base_link" />

  <joint name="base_joint" type="fixed">
    <origin xyz="0 0 0" rpy="0 0 0"/>
    <parent link="base_link" />
    <child link="J1_base/INPUT_INTERFACE"/>
  </joint>

  <xacro:actuator type="X8_9" name="J1_base" child="shoulder_bracket"/>
  <xacro:bracket type="X5HeavyRightOutside" name="shoulder_bracket" child="J2_shoulder"/>
  <xacro:actuator type="X8_16" name="J2_shoulder" child="shoulder_elbow"/>
  <xacro:link type="X5" extension="0.325" twist="${pi}" name="shoulder_elbow" child="J3_elbow"/>
  <xacro:actuator type="X8_9" name="J3_elbow" child="elbow_wrist1"/>
  <xacro:link type="X5" extension="0.325" twist="${pi}" name="elbow_wrist1" child="J4_wrist1"/>
  <xacro:actuator type="X5_1" name="J4_wrist1" child="wrist2_bracket"/>
  <xacro:bracket type="X5LightRight" name="wrist2_bracket" child="J5_wrist2"/>
  <xacro:actuator type="X5_1" name="J5_wrist2" child="end_effector"/>
  <xacro:gripper type="Custom" name="end_effector" mass="0.005"/>

</robot>

To be compatible with the hebi_description package standards, the URDF will be saved as <robot_name>.urdf.xacro in the urdf/kits folder.

You can visualize the arm URDF using RViz. To do so, run the following command.

ros2 launch hebi_description view_arm.launch.py hebi_arm:=<your_robot_name>

NOTE: Do not forget to build your workspace and source your setup before running the above command.

If the launch is successful, you should see a RViz window, with the arm and a GUI to control the joints of the arm.

ROS2 Control

To control HEBI arm using ros2_control, you need additional packages.

git clone https://github.com/HebiRobotics/hebi_hardware.git
git clone -b $ROS_DISTRO https://github.com/HebiRobotics/hebi_bringup.git # ROS_DISTRO can be either humble, iron, or jazzy

Install dependencies:

sudo apt install ros-$ROS_DISTRO-ros2-control ros-$ROS_DISTRO-ros2-controllers

You will also need three more files apart from the URDF file describing the arm:

  • ROS2 Control macro file
  • Xacro file comprising the ROS 2 Control macro file and the original URDF file
  • ROS2 Control parameter file

Both of these files should be placed in urdf/kits/ros2_control folder of the hebi_decription package.

ROS2 Control Macro File

The template for a HEBI Arm ROS2 Control Macro file is as follows:

<?xml version="1.0" encoding="UTF-8"?>
<robot xmlns:xacro="http://wiki.ros.org/xacro">

  <xacro:macro name="<your_robot_name>_ros2_control" params="
                name
                use_mock_hardware:=^|false
                mock_sensor_commands:=^|false
                sim_gazebo:=^|false
                families
                names
                hrdf_pkg
                hrdf_file
                gains_pkg
                gains_file"
               >

    <ros2_control name="${name}" type="system">

      <hardware>
        <xacro:if value="${use_mock_hardware}">
          <plugin>mock_components/GenericSystem</plugin>
          <param name="mock_sensor_commands">${mock_sensor_commands}</param>
        </xacro:if>
        <xacro:if value="${sim_gazebo}">
          <plugin>gazebo_ros2_control/GazeboSystem</plugin>
        </xacro:if>
        <xacro:unless value="${use_mock_hardware or sim_gazebo}">
          <param name="families">${families}</param>
          <param name="names">${names}</param>
          <param name="hrdf_pkg">${hrdf_pkg}</param>
          <param name="hrdf_file">${hrdf_file}</param>
          <param name="gains_pkg">${gains_pkg}</param>
          <param name="gains_file">${gains_file}</param>
          <plugin>hebi_hardware/HEBIHardwareInterface</plugin>
        </xacro:unless>
      </hardware>
      <joint name="joint_1">
        <command_interface name="position" />
        <state_interface name="position">
          <param name="initial_value">0.0</param>
        </state_interface>
        <state_interface name="velocity" />
      </joint>
      ...
      ...
      ...
      <joint name="joint_n">
        <command_interface name="position" />
        <state_interface name="position">
          <param name="initial_value">0.0</param>
        </state_interface>
        <state_interface name="velocity" />
      </joint>
    </ros2_control>

    <!-- Gazebo Classic plugins -->
    ...

    <!-- Gazebo plugins -->
    ...

  </xacro:macro>
</robot>

Name this file <your_robot_name>_macro.ros2_control.xacro.

NOTE: The gazebo classic and ignition plugins sections differ with each ROS version. Please refer to example files provided.

Combined URDF Xacro File

The template for the combined URDF xacro file is given below:

<?xml version="1.0" encoding="UTF-8"?>
<robot xmlns:xacro="http://wiki.ros.org/xacro" name="<your_robot_name>">

  <xacro:arg name="families" default="" />
  <xacro:arg name="names" default="" />
  <xacro:arg name="hrdf_pkg" default="" />
  <xacro:arg name="hrdf_file" default="" />
  <xacro:arg name="gains_pkg" default="" />
  <xacro:arg name="gains_file" default="" />

  <xacro:arg name="use_mock_hardware" default="false" />
  <xacro:arg name="mock_sensor_commands" default="false" />
  <xacro:arg name="sim_gazebo" default="false" />

  <xacro:include filename="/path/to/ros2_control_macro_file"/>

  <!-- create link fixed to the "world" -->
  <link name="world" />

  <joint name="world_to_base_joint" type="fixed">
    <origin xyz="0 0 0" rpy="0 0 0"/>
    <parent link="world" />
    <child link="base_link"/>
  </joint>

  <xacro:include filename="/path/to/original/URDF_file"/>
  
  <xacro:<your_robot_name>_ros2_control
    name="<your_robot_name>"
    use_mock_hardware="$(arg use_mock_hardware)"
    mock_sensor_commands="$(arg mock_sensor_commands)"
    sim_gazebo="$(arg sim_gazebo)"
    families="$(arg families)"
    names="$(arg names)"
    hrdf_pkg="$(arg hrdf_pkg)"
    hrdf_file="$(arg hrdf_file)"
    gains_pkg="$(arg gains_pkg)"
    gains_file="$(arg gains_file)" />

</robot>

This file is named similarly to the original URDF file, i.e., <robot_name>.urdf.xacro, but saved in the urdf/kits/ros2_control folder.

ROS2 Control Parameter File

More details about the ROS2 Control Parameter file can be found in the ROS2 Control documentation.

The parameter file for A-2085-05 looks like this:

controller_manager:
  ros__parameters:
    update_rate: 100  # Hz

    joint_state_broadcaster:
      type: joint_state_broadcaster/JointStateBroadcaster

    forward_position_controller:
      type: forward_command_controller/ForwardCommandController

    hebi_arm_controller:
      type: joint_trajectory_controller/JointTrajectoryController

forward_position_controller:
  ros__parameters:
    joints:
      - J1_base
      - J2_shoulder
      - J3_elbow
      - J4_wrist1
      - J5_wrist2
    interface_name: position

hebi_arm_controller:
  ros__parameters:
    joints:
      - J1_base
      - J2_shoulder
      - J3_elbow
      - J4_wrist1
      - J5_wrist2
    
    command_interfaces:
      - position
    
    state_interfaces:
      - position
      - velocity

    state_publish_rate: 50.0 # Defaults to 50
    action_monitor_rate: 20.0 # Defaults to 20

    allow_partial_joints_goal: false # Defaults to false
    constraints:
      stopped_velocity_tolerance: 0.01 # Defaults to 0.01
      goal_time: 0.0 # Defaults to 0.0 (start immediately)

To execute the ROS 2 Control node with the hardware, run the following command:

ros2 launch hebi_bringup bringup_arm.launch.py hebi_arm:=<your_robot_name> families:="<string_families_separated_by_semicolon>" names:="<string_names_separated_by_semicolon>"
use_mock_hardware:=true/false # Default is true

Here is an example for A-2085-05:

ros2 launch hebi_bringup bringup_arm.launch.py hebi_arm:=A-2085-05 families:="HEBI" names:="J1_base;J2_shoulder;J3_elbow;J4_wrist1;J5_wrist2"

There are many arguments that can be passed to the bringup_arm.launch.py file. Please refer the launch file to see all the parameters.

Gazebo Classic

To execute the ROS 2 Control node with Gazebo classic simulation, run the following command:

ros2 launch hebi_bringup bringup_arm_gazebo_classic.launch.py hebi_arm:=<your_robot_name>

NOTE: Do not forget to build your workspace and source your setup before running the above commands. Also, ensure you have necessary packages installed such as gazebo_ros, gazebo_ros2_control

To test the controllers, you can do so by running the following command:

ros2 launch hebi_bringup test_joint_trajectory_controller.launch.py

You should see the arm moving according to the joint positions given in the configuration file.

Gazebo (Ignition)

To execute the ROS 2 Control node with Gazebo (ignition) simulation, run the following command:

ros2 launch hebi_bringup bringup_arm_gazebo.launch.py hebi_arm:=<your_robot_name>

NOTE: Do not forget to build your workspace and source your setup before running the above commands. Also, ensure you have necessary packages installed such as ros_gz, ign_ros2_control / gz_ros2_control.

You can use the same launch file test_joint_trajectory_controller.launch.py to test the controller.

MoveIt

To use MoveIt with HEBI Arm, you need the ROS 2 control repositories and also the following:

git clone -b ros2 https://github.com/HebiRobotics/hebi_moveit_configs.git

Additionall, you will need to create a MoveIt configuration package for your arm. The steps to do so are given below.

Create MoveIt Configuration Package

Open MoveIt Setup Assistant using the following command:

ros2 run moveit_setup_assistant moveit_setup_assistant 

The following window will open:

Click on the Create New MoveIt Configuration Package button and load the URDF xacro file created in the first step, from the folder urdf/kits of hebi_description package. Finally, click on the Load Files button.

Once the files are loaded, you will see the following window:

Click on Self-Collisions on the sidebar, and click on Generate Collision Matrix button after choosing appropriate Sampling Density. This will generate the collision matrix for your arm.

Click on Virtual Joints on the sidebar, and click on Add Virtual Joint button to add a virtual joint from world to the base_link of your arm. Name the virtual joint as world_joint. Choose the Child Link as base_link and type world in the Parent Frame Name. Keep the Joint Type as fixed, and click on Save button. Finally, you will see the following window:

Click on Planning Groups on the sidebar, and click on Add Group button to add a planning group for your arm. Name the planning group as hebi_arm. Choose the kinematics solver as desired. Leave the remaining fields as default.

Now, click on Add joints button to add the joints of your arm to the planning group. Select all of the actuator joints and click on the right arrow button to add them to the Planning Group. Finally, click "Save" and you will see the following window:

Click on Robot Poses on the sidebar, and click on Add Pose button to add a robot pose for your arm. Name the robot pose as home, and set the joint angles for that position. Finally, click "Save".

It is recommended to add another robot pose to test MoveIt later. We add one more robot pose test, and the final window looks like this:

Click on ROS 2 Controllers on the sidebar and click on Auto Add JointTrajectoryController Controllers For Each Planning Group button. This will add the necessary ROS 2 controllers for your arm.

Similarly, click on MoveIt Controllers on the sidebar and click on Auto Add FollowJointsTrajectory Controllers For Each Planning Group button. This will add the necessary MoveIt controllers for your arm.

Go to the Author Information tab and fill in the necessary details.

Click on the Configuration Files tab, and select the desired output directory. Note that the files will be placed directly in the selected directory, not in a new subdirectory, so choose a directory matching convention such as <your_robot_name>_moveit_config. Click on the Generate Package button to generate the MoveIt configuration package for your arm. Once the package is generated, you will see the following window:

Finally, click on Exit Setup Assistant button to exit the MoveIt Setup Assistant.

To enable the MoveIt to access the HEBI Hardware Interface, you will need to make the following changes in the <your_robot_name>_moveit_config package.

config/<your_robot_name>.ros2_control.xacro

After the line <xacro:property name="initial_positions" value="${load_yaml(initial_positions_file)['initial_positions']}"/>, add the following:

<xacro:property name="families" value="${load_yaml(initial_positions_file)['families']}"/>
<xacro:property name="names" value="${load_yaml(initial_positions_file)['names']}"/>
<xacro:property name="hrdf_pkg" value="${load_yaml(initial_positions_file)['hrdf_pkg']}"/>
<xacro:property name="hrdf_file" value="${load_yaml(initial_positions_file)['hrdf_file']}"/>
<xacro:property name="gains_pkg" value="${load_yaml(initial_positions_file)['gains_pkg']}"/>
<xacro:property name="gains_file" value="${load_yaml(initial_positions_file)['gains_file']}"/>
<xacro:property name="use_mock_hardware" value="${load_yaml(initial_positions_file)['use_mock_hardware']}"/>
<xacro:property name="use_gazebo" value="${load_yaml(initial_positions_file)['use_gazebo']}"/>

Replace the <hardware> tag with the following:

<hardware>
  <xacro:if value="${use_mock_hardware}">
    <plugin>mock_components/GenericSystem</plugin>
  </xacro:if>
  <xacro:if value="${use_gazebo}">
    <plugin>gazebo_ros2_control/GazeboSystem</plugin>
  </xacro:if>
  <xacro:unless value="${use_mock_hardware or use_gazebo}">
    <!-- Parameters to initialize the Components -->
    <param name="families">${families}</param>
    <param name="names">${names}</param>
    <param name="hrdf_pkg">${hrdf_pkg}</param>
    <param name="hrdf_file">${hrdf_file}</param>
    <param name="gains_pkg">${gains_pkg}</param>
    <param name="gains_file">${gains_file}</param>
    <plugin>hebi_hardware/HEBIHardwareInterface</plugin>
  </xacro:unless>
</hardware>

After the <ros2_control> tag, add the following:

<!-- Gazebo Classic plugins -->
<xacro:if value="${use_gazebo}">
  <gazebo>
    <plugin filename="libgazebo_ros2_control.so" name="gazebo_ros2_control">
      <parameters>$(find hebi_a-2085-06)/config/ros2_controllers.yaml</parameters>
    </plugin>
  </gazebo>
</xacro:if>

The final file for A-2085-05 looks like this: <link_to_github_file>

config/initial_positions.yaml

At the end of the file, add the following:

families: "<families_as_string_separated_by_semicolon>"

names: "<names_as_string_separated_by_semicolon>"

hrdf_pkg: "hebi_description"
hrdf_file: "config/hrdf/<your_robot_name>.hrdf"
gains_pkg: "hebi_description"
gains_file: "config/gains/<your_robot_name>_gains.xml"

use_mock_hardware: true
use_gazebo: false

To test the MoveIt configuration, run the following command:

ros2 launch <your_robot_name>_moveit_config demo.launch.py

NOTE: Do not forget to build your workspace and source your setup before running the above command.

If the launch is successful, a RViz window opens up with the arm and a GUI to control the arm that looks like this:

To check if the MoveIt is able to plan and execute the trajectories, click on the Planning tab on the left sidebar, and choose home as the GoalState. Click on Plan and Execute button. This will plan and execute the trajectory to the home position, and you should see the arm move to the home position.

To test the MoveIt configuration with the hardware, change the use_mock_hardware to false in the initial_positions.yaml file, and relaunch the demo.launch.py file. This will launch the MoveIt with the hardware interface. Now, you can plan and execute the trajectories with the hardware. NOTE: Do not forget to build your workspace and source your setup before relaunching.