mirror of
https://github.com/huggingface/lerobot.git
synced 2026-05-18 10:10:08 +00:00
462a8aea90
Add first-class LeRobot support for the Seeed Studio reBot arm, replacing the out-of-tree `lerobot-robot-seeed-b601` / `lerobot-teleoperator-rebot-arm-102` plugin packages. New devices: - robot `rebot_b601_follower` — single-arm B601-DM follower (6-DOF + gripper, Damiao CAN motors via `motorbridge`) - robot `bi_rebot_b601_follower` — bimanual follower composing two single arms - teleoperator `rebot_102_leader` — single-arm StarArm102 / reBot Arm 102 leader (FashionStar UART servos via `motorbridge-smart-servo`) - teleoperator `bi_rebot_102_leader` — bimanual leader composing two single arms The bimanual variants reuse the single-arm classes and namespace each arm's observation/action keys with `left_` / `right_` prefixes, so a bimanual StarArm102 leader can teleoperate a bimanual reBot B601 follower. Optional SDK imports are guarded; a `rebot` extra installs `motorbridge` and `motorbridge-smart-servo`. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
110 lines
3.8 KiB
Python
110 lines
3.8 KiB
Python
#!/usr/bin/env python
|
|
|
|
# Copyright 2026 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.
|
|
|
|
from unittest.mock import MagicMock, patch
|
|
|
|
import pytest
|
|
|
|
from lerobot.teleoperators.bi_rebot_102_leader import BiRebotArm102Leader, BiRebotArm102LeaderConfig
|
|
from lerobot.teleoperators.rebot_102_leader import (
|
|
RebotArm102Leader,
|
|
RebotArm102LeaderConfig,
|
|
RebotArm102LeaderTeleopConfig,
|
|
)
|
|
|
|
_MODULE = "lerobot.teleoperators.rebot_102_leader.rebot_102_leader"
|
|
|
|
|
|
def _make_bus_mock(joint_ids: dict[str, int]) -> MagicMock:
|
|
bus = MagicMock(name="FashionStarServoMock")
|
|
bus.ping.return_value = True
|
|
|
|
def _sync_monitor(ids):
|
|
# Report each servo at 5 degrees raw.
|
|
monitors = {}
|
|
for servo_id in ids:
|
|
monitor = MagicMock()
|
|
monitor.angle_deg = 5.0
|
|
monitors[servo_id] = monitor
|
|
return monitors
|
|
|
|
bus.sync_monitor.side_effect = _sync_monitor
|
|
return bus
|
|
|
|
|
|
@pytest.fixture
|
|
def leader():
|
|
cfg = RebotArm102LeaderTeleopConfig(port="/dev/null")
|
|
bus_mock = _make_bus_mock(cfg.joint_ids)
|
|
with (
|
|
patch(f"{_MODULE}.require_package", lambda *a, **kw: None),
|
|
patch(f"{_MODULE}.FashionStarServo", return_value=bus_mock),
|
|
):
|
|
teleop = RebotArm102Leader(cfg)
|
|
teleop.connect(calibrate=False)
|
|
yield teleop
|
|
if teleop.is_connected:
|
|
teleop.disconnect()
|
|
|
|
|
|
def test_action_features_match_joints():
|
|
with patch(f"{_MODULE}.require_package", lambda *a, **kw: None):
|
|
teleop = RebotArm102Leader(RebotArm102LeaderTeleopConfig(port="/dev/null"))
|
|
assert set(teleop.action_features) == {f"{m}.pos" for m in teleop.motor_names}
|
|
assert teleop.feedback_features == {}
|
|
|
|
|
|
def test_invalid_config_rejected():
|
|
with patch(f"{_MODULE}.require_package", lambda *a, **kw: None):
|
|
bad = RebotArm102LeaderTeleopConfig(port="/dev/null", joint_directions={"shoulder_pan": 1})
|
|
with pytest.raises(ValueError, match="joint_directions"):
|
|
RebotArm102Leader(bad)
|
|
|
|
|
|
def test_connect_disconnect(leader):
|
|
assert leader.is_connected
|
|
leader.disconnect()
|
|
assert not leader.is_connected
|
|
|
|
|
|
def test_get_action_applies_direction_and_clamp(leader):
|
|
action = leader.get_action()
|
|
assert set(action) == {f"{m}.pos" for m in leader.motor_names}
|
|
# shoulder_pan has direction -1, so a +5deg raw reading flips to -5deg.
|
|
assert action["shoulder_pan.pos"] == pytest.approx(-5.0)
|
|
# Every joint stays within its configured range.
|
|
for motor, value in action.items():
|
|
lo, hi = leader.config.joint_ranges[motor.removesuffix(".pos")]
|
|
assert lo <= value <= hi
|
|
|
|
|
|
def test_send_feedback_not_implemented(leader):
|
|
with pytest.raises(NotImplementedError):
|
|
leader.send_feedback({})
|
|
|
|
|
|
def test_bimanual_prefixes_features():
|
|
with patch(f"{_MODULE}.require_package", lambda *a, **kw: None):
|
|
cfg = BiRebotArm102LeaderConfig(
|
|
left_arm_config=RebotArm102LeaderConfig(port="/dev/null0"),
|
|
right_arm_config=RebotArm102LeaderConfig(port="/dev/null1"),
|
|
)
|
|
teleop = BiRebotArm102Leader(cfg)
|
|
assert any(k.startswith("left_") for k in teleop.action_features)
|
|
assert any(k.startswith("right_") for k in teleop.action_features)
|
|
assert "left_gripper.pos" in teleop.action_features
|
|
assert "right_gripper.pos" in teleop.action_features
|