mirror of
https://github.com/huggingface/lerobot.git
synced 2026-05-11 14:49:43 +00:00
ca87ccd941
* feat(scripts): lerobot-rollout * fix(rollout) require dataset in dagger + use duration too * fix(docs): dagger num_episodes * test(rollout): fix expectations * fix(rollout): features check * fix(rollout): device and task propagation + feature pos + warn fps + move rename_map config * docs(rollout): edit rename_map instructions * chore(rollout): multiple minor improvements * chore(rollout): address coments + minor improvements * fix(rollout): enable default * fix(tests): default value RTCConfig * fix(rollout): robot_observation_processor and notify_observation at policy frequency instead of interpolator rate Co-authored-by: Pepijn <138571049+pkooij@users.noreply.github.com> * fix(rollout): prevent relativeactions with sync inference engine Co-authored-by: Pepijn <138571049+pkooij@users.noreply.github.com> * fix(rollout): rtc reanchor to non normalized state Co-authored-by: Pepijn <138571049+pkooij@users.noreply.github.com> * fix(rollout): fixing the episode length to use hwc (#3469) also reducing default length to 5 minutes * feat(rollout): go back to initial position is now a config * fix(rollout): properly propagating video_files_size_in_mb to lerobot_dataset (#3470) * chore(rollout): note about dagger correction stage * chore(docs): update comments and docstring * fix(test): move rtc relative out of rollout module * fix(rollout): address the review comments --------- Co-authored-by: Pepijn <138571049+pkooij@users.noreply.github.com> Co-authored-by: Maxime Ellerbach <maxime.ellerbach@huggingface.co>
135 lines
5.0 KiB
Python
135 lines
5.0 KiB
Python
# !/usr/bin/env python
|
|
|
|
# Copyright 2025 The HuggingFace Inc. team. All rights reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
"""Run a trained EE-space policy on SO100 without recording (base rollout).
|
|
|
|
Uses the rollout engine's :class:`BaseStrategy` (autonomous execution,
|
|
no dataset) with :class:`SyncInferenceConfig` (inline policy call per
|
|
control tick). The custom observation/action processors convert between
|
|
joint space (robot hardware) and end-effector space (policy I/O) via
|
|
forward/inverse kinematics.
|
|
"""
|
|
|
|
from lerobot.cameras.opencv import OpenCVCameraConfig
|
|
from lerobot.configs import PreTrainedConfig
|
|
from lerobot.model.kinematics import RobotKinematics
|
|
from lerobot.processor import (
|
|
RobotProcessorPipeline,
|
|
observation_to_transition,
|
|
robot_action_observation_to_transition,
|
|
transition_to_observation,
|
|
transition_to_robot_action,
|
|
)
|
|
from lerobot.robots.so_follower import SO100Follower, SO100FollowerConfig
|
|
from lerobot.robots.so_follower.robot_kinematic_processor import (
|
|
ForwardKinematicsJointsToEE,
|
|
InverseKinematicsEEToJoints,
|
|
)
|
|
from lerobot.rollout import BaseStrategyConfig, RolloutConfig, build_rollout_context
|
|
from lerobot.rollout.inference import SyncInferenceConfig
|
|
from lerobot.rollout.strategies import BaseStrategy
|
|
from lerobot.types import RobotAction, RobotObservation
|
|
from lerobot.utils.process import ProcessSignalHandler
|
|
from lerobot.utils.utils import init_logging
|
|
|
|
FPS = 30
|
|
DURATION_SEC = 60
|
|
TASK_DESCRIPTION = "My task description"
|
|
HF_MODEL_ID = "<hf_username>/<model_repo_id>"
|
|
|
|
|
|
def main():
|
|
init_logging()
|
|
|
|
# Robot configuration — the rollout engine will connect it inside build_rollout_context.
|
|
camera_config = {"front": OpenCVCameraConfig(index_or_path=0, width=640, height=480, fps=FPS)}
|
|
robot_config = SO100FollowerConfig(
|
|
port="/dev/tty.usbmodem5A460814411",
|
|
id="my_awesome_follower_arm",
|
|
cameras=camera_config,
|
|
use_degrees=True,
|
|
)
|
|
|
|
# Kinematic solver: we need the motor-name list, so peek at the robot once.
|
|
# (The rollout engine owns the connected instance; we only use this for introspection.)
|
|
temp_robot = SO100Follower(robot_config)
|
|
motor_names = list(temp_robot.bus.motors.keys())
|
|
|
|
# NOTE: It is highly recommended to use the urdf in the SO-ARM100 repo:
|
|
# https://github.com/TheRobotStudio/SO-ARM100/blob/main/Simulation/SO101/so101_new_calib.urdf
|
|
kinematics_solver = RobotKinematics(
|
|
urdf_path="./SO101/so101_new_calib.urdf",
|
|
target_frame_name="gripper_frame_link",
|
|
joint_names=motor_names,
|
|
)
|
|
|
|
# Joint-space observation → EE-space observation (consumed by the policy).
|
|
robot_joints_to_ee_pose_processor = RobotProcessorPipeline[RobotObservation, RobotObservation](
|
|
steps=[ForwardKinematicsJointsToEE(kinematics=kinematics_solver, motor_names=motor_names)],
|
|
to_transition=observation_to_transition,
|
|
to_output=transition_to_observation,
|
|
)
|
|
|
|
# EE-space action (produced by the policy) → joint-space action (sent to robot).
|
|
robot_ee_to_joints_processor = RobotProcessorPipeline[tuple[RobotAction, RobotObservation], RobotAction](
|
|
steps=[
|
|
InverseKinematicsEEToJoints(
|
|
kinematics=kinematics_solver,
|
|
motor_names=motor_names,
|
|
initial_guess_current_joints=True,
|
|
),
|
|
],
|
|
to_transition=robot_action_observation_to_transition,
|
|
to_output=transition_to_robot_action,
|
|
)
|
|
|
|
# Policy config (full model is loaded inside build_rollout_context).
|
|
policy_config = PreTrainedConfig.from_pretrained(HF_MODEL_ID)
|
|
policy_config.pretrained_path = HF_MODEL_ID
|
|
|
|
cfg = RolloutConfig(
|
|
robot=robot_config,
|
|
policy=policy_config,
|
|
strategy=BaseStrategyConfig(),
|
|
inference=SyncInferenceConfig(),
|
|
fps=FPS,
|
|
duration=DURATION_SEC,
|
|
task=TASK_DESCRIPTION,
|
|
)
|
|
|
|
signal_handler = ProcessSignalHandler(use_threads=True)
|
|
|
|
# Pass the EE kinematic processors via kwargs; the defaults (identity) would
|
|
# otherwise skip the joint↔EE conversion and the policy would receive the
|
|
# wrong observation/action space.
|
|
ctx = build_rollout_context(
|
|
cfg,
|
|
signal_handler.shutdown_event,
|
|
robot_action_processor=robot_ee_to_joints_processor,
|
|
robot_observation_processor=robot_joints_to_ee_pose_processor,
|
|
)
|
|
|
|
strategy = BaseStrategy(cfg.strategy)
|
|
try:
|
|
strategy.setup(ctx)
|
|
strategy.run(ctx)
|
|
finally:
|
|
strategy.teardown(ctx)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
main()
|