As we've seen in previous tutorials, commanding robot motion is simple and straightforward. For example, the incremental motion of the arm can be commanded by:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
robot.arm.move_by(0.1)
robot.push_command()
time.sleep(2.0)
robot.stop()
The absolute motion can be commanded by:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
robot.arm.move_to(0.1)
robot.push_command()
time.sleep(2.0)
robot.stop()
In the above examples, we executed a time.sleep()
after robot.push_command()
. This allows the joint time to complete its motion. As an alternative, we can use the wait_until_at_setpoint()
method that polls the joint position versus the target position. We can also interrupt a motion by sending a new motion command at any time. For example, try the following script:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
#Move to full retraction
robot.arm.move_to(0.0)
robot.push_command()
robot.arm.wait_until_at_setpoint()
#Move to full extension
robot.arm.move_to(0.5)
robot.push_command()
#Interrupt motion midway and retract again
time.sleep(2.0)
robot.arm.move_to(0.0)
robot.arm.wait_until_at_setpoint()
robot.stop()
You will see the arm fully retract, begin to extend, and then fully retract again.
All joints support trapezoidal motion profile generation. Other types of controllers are available (splined trajectory, PID, velocity, etc) but they are not covered here. The trapezoidal motion controllers require three values:
We provide 'defaults' for the velocity and acceleration settings, as well as 'fast', and 'slow' settings. These values have been tuned to be appropriate for the safe movement of the robot. These values can be queried using the stretch_params.py
tool:
stretch_params.py | grep arm | grep motion | grep default
stretch_body.robot_params.nominal_params param.arm.motion.fast.accel_m 0.14
stretch_body.robot_params.nominal_params param.arm.motion.fast.vel_m 0.14
We see that the arm motion in 'default' mode will move with a velocity of 0.14 m/s and an acceleration of 0.14 m/s^2.
The move_by
and move_to
commands support optional motion profile parameters. For example, for a fast motion:
v = robot.arm.params['motion']['fast']['vel_m']
a = robot.arm.params['motion']['fast']['accel_m']
robot.arm.move_by(x_m=0.1, v_m=v, a_m=a)
robot.push_command()
The motion will use the 'default' motion profile settings if no values are specified.
All joints obey motion limits which are specified in the robot parameters.
stretch_params.py | grep arm | grep range_m
stretch_user_params.yaml param.arm.range_m [0.0, 0.515]
These are the mechanical limits of the joints and have been set at the factory to prevent damage to the hardware. It is not recommended to set them to be greater than the factory-specified values. However, they can be further limited if desired by setting soft motion limits:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
robot.arm.set_soft_motion_limit_min(0.2)
robot.arm.set_soft_motion_limit_max(0.4)
#Will move to position 0.2
robot.arm.move_to(0.0)
robot.push_command()
robot.arm.wait_until_at_setpoint()
#Will move to position 0.4
robot.arm.move_to(0.5)
robot.push_command()
robot.arm.wait_until_at_setpoint()
robot.stop()
The above examples have focused on the motion of the arm. Like the lift and the base, the arm utilizes Hello Robot's custom stepper motor controller. Control of the Dynamixels of the head and the end-of-arm is very similar to that of the arm, though not identical.
As we see here, the robot.push_command()
call is not required as the motion begins instantaneously and is not queued. In addition, the Dynamixel servos are managed as a chain of devices, so we must pass in the joint name along with the command.
import stretch_body.robot
from stretch_body.hello_utils import deg_to_rad
robot=stretch_body.robot.Robot()
robot.startup()
robot.head.move_to('head_pan',0)
robot.head.move_to('head_tilt',0)
time.sleep(3.0)
robot.head.move_to('head_pan',deg_to_rad(90.0))
robot.head.move_to('head_tilt',deg_to_rad(45.0))
time.sleep(3.0)
robot.stop()
Similar to the stepper joints, the Dynamixel joints accept motion profile and motion limit commands. For example, here we restrict the head pan range of motion while executing both a fast and slow move:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
#Limit range of motion of head_pan
robot.head.get_motor('head_pan').set_soft_motion_limit_min(deg_to_rad(-45.0))
robot.head.get_motor('head_pan').set_soft_motion_limit_max(deg_to_rad(45.0))
#Do a fast motion
v = robot.params['head_pan']['motion']['fast']['vel']
a = robot.params['head_pan']['motion']['fast']['accel']
robot.head.move_to('head_pan',deg_to_rad(-90.0),v_r=v, a_r=a)
time.sleep(3.0)
#Do a slow motion
v = robot.params['head_pan']['motion']['slow']['vel']
a = robot.params['head_pan']['motion']['slow']['accel']
robot.head.move_to('head_pan',deg_to_rad(90.0),v_r=v, a_r=a)
time.sleep(3.0)
robot.stop()
The Base also supports a velocity control mode which can be useful with navigation planners. The Base controllers will automatically switch between velocity and position control. For example:
robot.base.translate_by(x_m=0.5)
robot.push_command()
time.sleep(4.0) #wait
robot.base.set_rotational_velocity(v_r=0.1) #switch to velocity controller
robot.push_command()
time.sleep(4.0) #wait
robot.base.set_rotational_velocity(v_r=0.0) #stop motion
robot.push_command()
!!! warning As shown, care should be taken to set commanded velocities to zero on exit to avoid runaway.
Most users will control robot motion using the move_to
and move_by
commands as described above. However, there are numerous other low-level controller modes available. While this is a topic for advanced users, it is worth noting that each joint has a default safety mode and a default position control mode. These are:
Joint | Default Safety Mode | Default Position Control Mode |
---|---|---|
left_wheel | Freewheel | Trapezoidal position control |
right_wheel | Freewheel | Trapezoidal position control |
lift | Gravity compensated 'float' | Trapezoidal position control |
arm | Freewheel | Trapezoidal position control |
head_pan | Torque disabled | Trapezoidal position control |
head_tilt | Torque disabled | Trapezoidal position control |
wrist_yaw | Torque disabled | Trapezoidal position control |
stretch_gripper | Torque disabled | Trapezoidal position control |
Each joint remains in its Safety Mode
when no program is running. When the <device>.startup()
function is called, the joint controller transitions from Safety Mode
to its Default Position Control Mode
. It is then placed back in Safety Mode
when <device>.stop()
is called.
Runstop activation will cause the Base, Arm, and Lift to switch to Safety Mode
and subsequent motion commands will be ignored. The motion commands will resume smoothly when the Runstop is deactivated. This is usually done via the Runstop button. However, it can also be done via the Pimu interface. For example:
import stretch_body.robot
robot=stretch_body.robot.Robot()
robot.startup()
#Move to full retraction
robot.arm.move_to(0.0)
robot.push_command()
robot.arm.wait_until_at_setpoint()
#Move to full extension
robot.arm.move_to(0.5)
robot.push_command()
#Runstop motion midway through the motion
input('Hit enter to runstop motion')
robot.pimu.runstop_event_trigger()
robot.push_command()
input('Hit enter to restart motion')
robot.pimu.runstop_event_reset()
robot.push_command()
robot.arm.move_to(0.5)
robot.push_command()
robot.arm.wait_until_at_setpoint()
robot.stop()
The Arm, Lift, and Base support a guarded motion function. It will automatically transition the actuator from Control mode to Safety mode when the exerted motor torque exceeds a threshold.
This functionality is most useful for the Lift and the Arm. It allows these joints to safely stop upon contact. It can be used to:
For more information on guarded motion, see the Contact Models Tutorial
The Arm, Lift, and Base actuators have a hardware synchronization mechanism. This allows for stepper controller commands to be time synchronized across joints. This behavior can be disabled via the user YAML. By default the settings are:
stretch_params.py | grep enable_sync_mode
stretch_body.robot_params.nominal_params param.hello-motor-arm.gains.enable_sync_mode 1
stretch_body.robot_params.nominal_params param.hello-motor-left-wheel.gains.enable_sync_mode 1
stretch_body.robot_params.nominal_params param.hello-motor-lift.gains.enable_sync_mode 1
stretch_body.robot_params.nominal_params param.hello-motor-right-wheel.gains.enable_sync_mode 1
It can be useful to poll the status of a joint during a motion to modify the robot's behavior, etc. The useful status values include:
robot.arm.status['pos'] #Joint position
robot.arm.status['vel'] #Joint velocity
robot.arm.motor.status['effort_pct'] #Joint effort (-100 to 100) (derived from motor current)
robot.arm.motor.status['near_pos_setpoint'] #Is sensed position near commanded position
robot.arm.motor.status['near_vel_setpoint'] #Is sensed velocity near commanded velocity
robot.arm.motor.status['is_moving'] #Is the joint in motion
robot.arm.motor.status['in_guarded_event'] #Has a guarded event occured
robot.arm.motor.status['in_safety_event'] #Has a safety event occured
The following update rates apply to Stretch:
Item | Rate | Notes |
---|---|---|
Status data for Arm, Lift, Base, Wacc, and Pimu | 25Hz | Polled automatically by Robot thread |
Status data for End of Arm and Head servos | 15Hz | Polled automatically by Robot thread |
Command data for Arm, Lift, Base, Wacc, Pimu | N/A | Commands are queued and executed upon calling robot.push_command( ) |
Command data for End of Arm and Head servos | N/A | Commands execute immediately |
!!! note Motion commands are non-blocking and it is the responsibility of the user code to poll the Robot Status to determine when and if a motion target has been achieved.
!!! info The Stretch Body interface is not designed to support high bandwidth control applications. The natural dynamics of the robot actuators do not support high bandwidth control, and the USB-based interface limits high-rate communication.
!!! tip In practice, a Python-based control loop that calls push_command() at 1Hz to 10Hz is sufficiently matched to the robot's natural dynamics.