Files
Uni-Lab-OS/docs/moveit2_integration_summary.md

59 KiB
Raw Blame History

Uni-Lab-OS MoveIt2 集成架构总结

概览

Uni-Lab-OS 通过三个核心文件实现了对 MoveIt2ROS 2 运动规划框架)的深度集成,形成了从底层运动规划接口业务逻辑封装场景构建与启动的完整链路。

┌──────────────────────────────────────────────────────────────────────┐
│                    resource_visalization.py                         │
│         场景构建层URDF/SRDF 生成、MoveIt2 节点启动                   │
│   ros2_control_node / move_group / robot_state_publisher / rviz2   │
└────────────────────────────┬─────────────────────────────────────────┘
                             │ 提供 MoveIt2 运行环境Planning Scene、控制器
                             ▼
┌──────────────────────────────────────────────────────────────────────┐
│                       moveit_interface.py                           │
│         业务逻辑层pick_and_place、set_position、set_status          │
│         管理多个 MoveGroup提供 FK/IK 计算、资源 TF 更新              │
└────────────────────────────┬─────────────────────────────────────────┘
                             │ 调用 MoveIt2 Python API
                             ▼
┌──────────────────────────────────────────────────────────────────────┐
│                          moveit2.py                                 │
│          底层接口层MoveIt2 ROS 2 Python 客户端实现                   │
│  Action Client / Service Client / Collision Scene / 轨迹执行         │
└──────────────────────────────────────────────────────────────────────┘

1. moveit2.py — MoveIt2 底层 Python 接口

路径: unilabos/devices/ros_dev/moveit2.py
行数: ~2443 行
角色: 对 MoveIt2 ROS 2 接口的完整 Python 封装,是整个运动控制的基础。

1.1 核心类:MoveIt2

对 MoveIt2 框架的 ROS 2 通信协议进行了全面的 Python 封装,提供了规划、执行、运动学计算和碰撞管理的统一接口。

1.1.1 ROS 2 通信拓扑

MoveIt2 类在初始化时创建了丰富的 ROS 2 通信端点:

类型 名称 用途
Subscriber /joint_states 实时获取关节状态BEST_EFFORT QoS
Action Client /move_action 通过 MoveGroup Action 一体化规划+执行
Action Client /execute_trajectory 独立的轨迹执行
Service Client /plan_kinematic_path 关节空间/笛卡尔空间运动规划
Service Client /compute_cartesian_path 笛卡尔路径规划(直线插补)
Service Client /compute_fk 正运动学计算
Service Client /compute_ik 逆运动学计算
Service Client /get_planning_scene 获取当前规划场景
Service Client /apply_planning_scene 应用修改后的规划场景
Publisher /collision_object 发布碰撞物体
Publisher /attached_collision_object 发布附着碰撞物体
Publisher /trajectory_execution_event 发送轨迹取消指令

1.1.2 运动规划与执行

提供两种执行模式:

  • MoveGroup Action 模式 (use_move_group_action=True):规划和执行合并在一个 Action 调用中完成,由 MoveIt2 内部管理整个流程。
  • Plan + Execute 模式:先通过 Service 调用进行路径规划(/plan_kinematic_path/compute_cartesian_path),再通过 ExecuteTrajectory Action 执行。

关键方法:

方法 说明
move_to_pose() 移动到目标位姿(位置 + 四元数姿态),支持笛卡尔直线规划
move_to_configuration() 移动到目标关节配置
plan() / plan_async() 异步路径规划,返回 JointTrajectory
execute() 执行规划好的轨迹
wait_until_executed() 阻塞等待运动完成,返回成功/失败
cancel_execution() 取消当前运动

1.1.3 目标约束系统

支持多层次的目标设定:

  • 位置约束 (set_position_goal): 以球形区域定义目标位置的容差
  • 姿态约束 (set_orientation_goal): 支持 Euler 角和旋转向量两种参数化方式
  • 关节约束 (set_joint_goal): 直接指定目标关节角度
  • 复合约束 (set_pose_goal): 位置 + 姿态的组合
  • 路径约束 (set_path_joint_constraint, set_path_position_constraint, set_path_orientation_constraint): 对整条运动路径施加约束
  • 多目标组 (create_new_goal_constraint): 支持同时设置多组目标约束

1.1.4 运动学服务

  • 正运动学 (FK): compute_fk() / compute_fk_async() — 给定关节角求末端位姿
  • 逆运动学 (IK): compute_ik() / compute_ik_async() — 给定目标位姿求关节角,支持传入约束和避碰选项

1.1.5 碰撞场景管理

提供完整的 Planning Scene 管理接口:

方法 说明
add_collision_box() 添加长方体碰撞体
add_collision_sphere() 添加球形碰撞体
add_collision_cylinder() 添加圆柱碰撞体
add_collision_cone() 添加锥形碰撞体
add_collision_mesh() 添加三角网格碰撞体(依赖 trimesh
move_collision() 移动已有碰撞体
remove_collision_object() 移除碰撞体
attach_collision_object() 将碰撞体附着到机器人 link 上
detach_collision_object() 从机器人上分离碰撞体
allow_collisions() 修改 Allowed Collision Matrix允许/禁止特定碰撞
clear_all_collision_objects() 清空所有碰撞物体

1.1.6 状态管理

通过 MoveIt2State 枚举和线程锁实现并发安全的状态管理:

  • IDLE: 空闲
  • REQUESTING: 已发送运动请求,等待接受
  • EXECUTING: 正在执行轨迹

配合 ignore_new_calls_while_executing 标志,防止运动冲突。

1.1.7 辅助工具函数

文件末尾提供模块级工具函数:

  • init_joint_state(): 构造 JointState 消息
  • init_execute_trajectory_goal(): 构造 ExecuteTrajectory.Goal 消息
  • init_dummy_joint_trajectory_from_state(): 构造用于重置控制器的虚拟轨迹

2. moveit_interface.py — 业务逻辑封装层

路径: unilabos/devices/ros_dev/moveit_interface.py
行数: 385 行
角色: 将 MoveIt2 底层接口封装为实验室场景中可用的高级操作pick/place、状态切换、资源管理

2.1 核心类:MoveitInterface

MoveitInterface 是连接实验室业务逻辑与 MoveIt2 运动规划的桥梁。它不直接继承 MoveIt2,而是通过组合的方式管理多个 MoveIt2 实例(每个 MoveGroup 一个),并在其上构建抓取放置、状态切换等实验室级操作。


2.1.1 类属性与实例属性

class MoveitInterface:
    _ros_node: BaseROS2DeviceNode    # ROS 2 节点引用post_init 注入)
    tf_buffer: Buffer                # TF2 坐标变换缓冲区
    tf_listener: TransformListener   # TF2 变换监听器

构造函数参数:

参数 类型 说明
moveit_type str 设备类型标识,用于定位 device_mesh/devices/{moveit_type}/config/move_group.json 配置文件
joint_poses dict 预定义关节位姿字典,结构为 {move_group: {status_name: [joint_values...]}}
rotation optional 设备旋转参数(预留)
device_config optional 设备自定义配置

实例属性:

属性 类型 初始值 说明
data_config dict 从 JSON 加载 Move Group 配置,结构为 {group_name: {base_link_name, end_effector_name, joint_names}}
arm_move_flag bool False 机械臂运动标志(预留)
move_option list ["pick", "place", "side_pick", "side_place"] 支持的抓取/放置动作类型
joint_poses dict 构造函数传入 预定义关节位姿查找表
cartesian_flag bool False 当前是否使用笛卡尔直线规划(在 pick_and_place 执行过程中动态切换)
mesh_group list ["reactor", "sample", "beaker"] 碰撞网格分组类别
moveit2 dict {} MoveIt2 实例字典,键为 Move Group 名称
resource_action str/None None 发现的 tf_update Action Server 名称
resource_client ActionClient/None None 用于发送资源 TF 更新请求的 Action Client
resource_action_ok bool False tf_update Action Server 是否就绪

2.1.2 初始化流程:__init__ + post_init

采用两阶段初始化设计:

阶段一:__init__(纯数据初始化,不依赖 ROS 2

__init__(moveit_type, joint_poses, ...)
    │
    ├── 加载 move_group.json 配置文件
    │   路径: device_mesh/devices/{moveit_type}/config/move_group.json
    │   内容: { "arm": { "base_link_name": "base_link",
    │                     "end_effector_name": "tool0",
    │                     "joint_names": ["joint1", "joint2", ...] } }
    │
    ├── 初始化状态标志 (arm_move_flag, cartesian_flag, resource_action_ok)
    └── 记录 joint_poses 预定义位姿

阶段二:post_init(ros_node)ROS 2 依赖初始化)

post_init(ros_node)
    │
    ├── 保存 ROS 节点引用 → self._ros_node
    │
    ├── 创建 TF2 基础设施
    │   ├── Buffer() → self.tf_buffer
    │   └── TransformListener(buffer, node) → self.tf_listener
    │
    ├── 遍历 data_config 中的每个 Move Group
    │   │
    │   │  对于每个 {move_group, config}:
    │   │
    │   ├── 生成带设备前缀的名称
    │   │   ├── base_link_name  = "{device_id}_{config.base_link_name}"
    │   │   ├── end_effector    = "{device_id}_{config.end_effector_name}"
    │   │   └── joint_names     = ["{device_id}_{name}" for name in config.joint_names]
    │   │
    │   └── 创建 MoveIt2 实例
    │       self.moveit2[move_group] = MoveIt2(
    │           node=ros_node,
    │           joint_names=...,
    │           base_link_name=...,
    │           end_effector_name=...,
    │           group_name="{device_id}_{move_group}",    ← MoveIt2 Planning Group 名
    │           callback_group=ros_node.callback_group,   ← 共享回调组
    │           use_move_group_action=True,                ← 使用 MoveGroup Action 模式
    │           ignore_new_calls_while_executing=True      ← 防止运动冲突
    │       )
    │       .allowed_planning_time = 3.0                  ← 规划超时 3 秒
    │
    └── 创建定时器 (1秒间隔)
        → wait_for_resource_action()                      ← 异步等待 tf_update Action Server

设备名前缀机制所有关节名、link 名、group 名都加上 {device_id}_ 前缀。这是多设备共存的关键——当多台机械臂在同一 ROS 2 网络中运行时前缀保证名称唯一MoveIt2 能正确区分不同设备的规划组。


2.1.3 资源 TF 更新服务发现

MoveitInterface 需要在运行时动态更新实验室资源(如烧杯、样品瓶)的 TF 父链接——当机械臂抓取物体时,物体的 TF 从 world 切换到末端执行器;放置时切换回 world

wait_for_resource_action() [定时器回调1秒间隔]
    │
    ├── 若 resource_action_ok 已为 True → 直接返回
    │
    ├── 轮询 check_tf_update_actions()
    │   │
    │   ├── 遍历所有 ROS 2 Topic
    │   ├── 查找 topic_type == "action_msgs/msg/GoalStatusArray" 的 Topic
    │   ├── 从 Topic 名称中提取 Action 名(去掉 "/_action/status" 后缀)
    │   └── 检查最后一段是否为 "tf_update" → 返回 Action 名称
    │
    ├── 创建 ActionClient(SendCmd, action_name)
    ├── 等待 Action Server 就绪 (timeout=5s循环等待)
    └── 设置 resource_action_ok = True

为什么用 Topic 发现而非硬编码?
tf_update Action Server 由 resource_mesh_manager 节点提供,其命名空间可能随部署配置变化。通过 Topic 自动发现机制,MoveitInterface 能自适应不同的部署环境。


2.1.4 底层运动方法:moveit_task

def moveit_task(self, move_group, position, quaternion,
                speed=1, retry=10, cartesian=False,
                target_link=None, offsets=[0,0,0])

这是笛卡尔空间运动的核心封装,所有高级方法最终都通过它调用 MoveIt2.move_to_pose()

执行流程:

moveit_task(move_group, position, quaternion, speed, retry, ...)
    │
    ├── 速度限幅: speed_ = clamp(speed, 0.1, 1.0)
    ├── 设置 MoveIt2 速度:
    │   ├── moveit2[group].max_velocity     = speed_
    │   └── moveit2[group].max_acceleration = speed_
    │
    ├── 计算最终位置: pose_result = position + offsets (逐元素相加)
    │
    └── 重试循环 (最多 retry+1 次):
        │
        ├── moveit2[group].move_to_pose(
        │       target_link=target_link,       ← 可指定非默认末端 link
        │       position=pose_result,           ← 目标 [x, y, z]
        │       quat_xyzw=quaternion,           ← 目标 [qx, qy, qz, qw]
        │       cartesian=cartesian,            ← 笛卡尔直线 or 自由空间
        │       cartesian_max_step=0.01,        ← 笛卡尔插补步长 1cm
        │       weight_position=1.0             ← 位置约束权重
        │   )
        │
        ├── re_ = moveit2[group].wait_until_executed()  ← 阻塞等待
        │
        └── 若 re_ 为 True → 返回成功; 否则 retry -= 1 继续

参数说明:

参数 说明 MoveIt2 对应
position 目标位置 [x, y, z](米) move_to_pose(position=...)
quaternion 目标姿态四元数 [x, y, z, w] move_to_pose(quat_xyzw=...)
speed 速度因子 0.1~1.0,同时控制速度和加速度 max_velocity + max_acceleration
retry 规划失败时的最大重试次数 N/A应用层重试
cartesian 是否使用笛卡尔直线规划 move_to_pose(cartesian=...)
target_link 目标 link默认末端执行器 move_to_pose(target_link=...)
offsets 位置偏移量 [dx, dy, dz] 叠加到 position

2.1.5 底层运动方法:moveit_joint_task

def moveit_joint_task(self, move_group, joint_positions,
                      joint_names=None, speed=1, retry=10)

关节空间运动的核心封装,调用 MoveIt2.move_to_configuration()

执行流程:

moveit_joint_task(move_group, joint_positions, joint_names, speed, retry)
    │
    ├── 关节角度转 float: joint_positions_ = [float(x) for x in joint_positions]
    ├── 速度限幅: speed_ = clamp(speed, 0.1, 1.0)
    ├── 设置 MoveIt2 速度
    │
    └── 重试循环:
        │
        ├── moveit2[group].move_to_configuration(
        │       joint_positions=joint_positions_,
        │       joint_names=joint_names        ← None 时使用 MoveIt2 默认关节名
        │   )
        │
        ├── re_ = moveit2[group].wait_until_executed()
        │
        ├── 打印 FK 结果 (调试用):
        │   compute_fk(joint_positions) → 显示对应的末端位姿
        │
        └── 若成功 → 返回; 否则 retry -= 1

moveit_task 的区别:

  • moveit_task:目标是笛卡尔空间位姿(位置 + 姿态MoveIt2 自动进行 IK 求解
  • moveit_joint_task:目标直接是关节角度,无需 IK 计算,确定性更高
  • 每次循环后调用 compute_fk 输出当前末端位姿,便于调试

2.1.6 资源 TF 管理:resource_manager

def resource_manager(self, resource, parent_link)

通过 SendCmd Action 向 tf_update 服务发送 TF 父链接更新请求。

resource_manager("beaker_1", "tool0")
    │
    ├── 构造 SendCmd.Goal:
    │   goal.command = '{"beaker_1": "tool0"}'    ← JSON 格式: {资源名: 新父link}
    │
    └── resource_client.send_goal(goal)           ← 异步发送,不等待结果

在 pick/place 流程中的角色:

  • pick 时resource_manager(resource, end_effector_name) — 资源跟随末端执行器
  • pick 且有 target 时resource_manager(resource, target) — 资源挂到指定 link
  • place 时resource_manager(resource, "world") — 资源释放到世界坐标系

2.1.7 直接位姿控制:set_position

def set_position(self, command: str)

最简单的运动接口,解析 JSON 指令后直接委托给 moveit_task

JSON 指令格式:

{
    "position": [0.3, 0.0, 0.5],
    "quaternion": [0.0, 0.0, 0.0, 1.0],
    "move_group": "arm",
    "speed": 0.5,
    "retry": 10
}

调用链:

set_position(command_json)
    │
    ├── JSON 解析 (替换单引号为双引号)
    └── moveit_task(**cmd_dict)
        └── MoveIt2.move_to_pose(...)

2.1.8 预定义状态切换:set_status

def set_status(self, command: str)

将机械臂移动到预定义的关节配置(如 home 位、准备位等),关节角度从 self.joint_poses 查找表中获取。

JSON 指令格式:

{
    "status": "home",
    "move_group": "arm",
    "speed": 0.8,
    "retry": 5
}

调用链:

set_status(command_json)
    │
    ├── JSON 解析
    ├── 查找预定义关节角: joint_poses[move_group][status]
    │   例: joint_poses["arm"]["home"] → [0.0, -1.57, 1.57, 0.0, 0.0, 0.0]
    │
    └── moveit_joint_task(move_group, joint_positions, speed, retry)
        └── MoveIt2.move_to_configuration(...)

joint_poses 查找表结构:

{
    "arm": {
        "home":    [0.0, -1.57, 1.57, 0.0, 0.0, 0.0],
        "ready":   [0.0, -0.78, 1.57, 0.0, 0.78, 0.0],
        "pick_A1": [0.5, -1.2, 1.0, 0.0, 0.5, 0.3],
        ...
    },
    "gripper": {
        "open":  [0.04],
        "close": [0.0],
        ...
    }
}

2.1.9 核心方法详解:pick_and_place

def pick_and_place(self, command: str)

这是 MoveitInterface 最复杂的方法,实现了完整的抓取-放置工作流。它动态构建一个有序函数列表 (function_list),然后顺序执行。

JSON 指令格式(完整参数):

{
    "option": "pick",                         // *必须: pick/place/side_pick/side_place
    "move_group": "arm",                      // *必须: MoveIt2 规划组名
    "status": "pick_station_A",               // *必须: 在 joint_poses 中的目标状态名
    "resource": "beaker_1",                   //  要操作的资源名称
    "target": "custom_link",                  //  pick 时资源附着的目标 link (默认末端执行器)
    "lift_height": 0.05,                      //  抬升高度 (米)
    "x_distance": 0.1,                        //  X 方向水平移动距离 (米)
    "y_distance": 0.0,                        //  Y 方向水平移动距离 (米)
    "speed": 0.5,                             //  运动速度因子 (0.1~1.0)
    "retry": 10,                              //  规划失败重试次数
    "constraints": [0, 0, 0, 0.5, 0, 0]       //  各关节约束容差 (>0 时生效)
}
阶段 1指令解析与动作类型判定
pick_and_place(command_json)
    │
    ├── JSON 解析
    ├── 动作类型判定:
    │   move_option = ["pick", "place", "side_pick", "side_place"]
    │                    0        1         2            3
    │   option_index = move_option.index(cmd["option"])
    │   place_flag   = option_index % 2    ← 0=pick类, 1=place类
    │
    ├── 提取运动参数:
    │   config = {speed, retry, move_group}  ← 从 cmd_dict 中按需提取
    │
    └── 获取目标关节位姿:
        joint_positions_ = joint_poses[move_group][status]
阶段 2构建资源 TF 更新动作
根据 place_flag 决定资源 TF 操作:

    if pick 类 (place_flag == 0):
        if "target" 已指定:
            function_list += [resource_manager(resource, target)]       ← 挂到自定义 link
        else:
            function_list += [resource_manager(resource, end_effector)] ← 挂到末端执行器
    
    if place 类 (place_flag == 1):
        function_list += [resource_manager(resource, "world")]         ← 释放到世界坐标
阶段 3构建关节约束
if "constraints" 存在于指令中:
    for i, tolerance in enumerate(constraints):
        if tolerance > 0:
            JointConstraint(
                joint_name = moveit2[group].joint_names[i],
                position   = joint_positions_[i],    ← 约束中心 = 目标关节角
                tolerance_above = tolerance,
                tolerance_below = tolerance,
                weight = 1.0
            )

约束的作用:限制 IK 求解的搜索空间,确保机械臂在抬升/移动过程中保持特定关节(如肘关节)在安全范围内。

阶段 4Alift_height 的完整流程

这是最复杂的场景,涉及 FK/IK 计算和多段运动拼接:

if "lift_height" 存在:
    │
    ├── Step 1: FK 计算 → 获取目标关节配置对应的末端位姿
    │   retval = compute_fk(joint_positions_)   ← 可能需要重试
    │   pose      = [retval.position.x, .y, .z]
    │   quaternion = [retval.orientation.x, .y, .z, .w]
    │
    ├── Step 2: 构建"下降到目标点"动作
    │   function_list = [moveit_task(position=pose, ...)] + function_list
    │   注:此时 function_list 已包含 resource_manager它被插入到中间
    │
    ├── Step 3: 构建"从目标点抬升"动作
    │   pose[2] += lift_height                 ← Z 轴抬升
    │   function_list += [moveit_task(position=pose_lifted, ...)]
    │
    ├── Step 4 (可选): 水平移动
    │   if "x_distance":
    │       deep_pose = copy(pose_lifted)
    │       deep_pose[0] += x_distance
    │       function_list = [moveit_task(pose_lifted)] + function_list
    │       function_list += [moveit_task(deep_pose)]
    │   elif "y_distance":
    │       类似处理 Y 方向
    │
    ├── Step 5: IK 预计算 → 将末端位姿转换为安全的关节配置
    │   retval_ik = compute_ik(
    │       position = end_pose,               ← 最终抬升/移动后的位姿
    │       quat_xyzw = quaternion,
    │       constraints = Constraints(joint_constraints=constraints)
    │   )
    │   position_ = 从 IK 结果提取各关节角度
    │
    └── Step 6: 构建"关节空间移动到起始位"动作
        function_list = [moveit_joint_task(position_)] + function_list
阶段 4Blift_height 的简单流程
else (无 lift_height):
    │
    └── 直接关节运动到目标位姿
        function_list = [moveit_joint_task(joint_positions_)] + function_list
阶段 5顺序执行动作列表
for i, func in enumerate(function_list):
    │
    ├── 设置规划模式:
    │   i == 0: cartesian_flag = False    ← 第一步用自由空间规划(大范围移动)
    │   i >  0: cartesian_flag = True     ← 后续用笛卡尔直线规划(精确控制)
    │
    ├── result = func()                   ← 执行动作
    │
    └── if not result:
        return failure                    ← 任一步骤失败即中止
完整 pick 流程示例(含 lift_height + x_distance

假设指令为pick beaker_1 from station_A, lift 5cm, move 10cm in X

最终 function_list 执行顺序:
┌─────────────────────────────────────────────────────────────────────┐
│ Step 0: moveit_joint_task → IK 求解的关节角                         │
│         [cartesian=False, 自由空间规划]                              │
│         机械臂从当前位置移动到抬升位                                   │
├─────────────────────────────────────────────────────────────────────┤
│ Step 1: moveit_task → 水平移动后的抬升位 (pose_lifted)               │
│         [cartesian=True, 笛卡尔直线]                                │
│         对齐到 station_A 正上方                                      │
├─────────────────────────────────────────────────────────────────────┤
│ Step 2: moveit_task → station_A 目标位姿 (FK 计算的位置)             │
│         [cartesian=True, 笛卡尔直线]                                │
│         末端执行器下降到目标点                                        │
├─────────────────────────────────────────────────────────────────────┤
│ Step 3: resource_manager("beaker_1", end_effector)                 │
│         资源 TF 附着到末端执行器                                      │
├─────────────────────────────────────────────────────────────────────┤
│ Step 4: moveit_task → 抬升位 (z + lift_height)                     │
│         [cartesian=True, 笛卡尔直线]                                │
│         抓取后垂直抬升                                               │
├─────────────────────────────────────────────────────────────────────┤
│ Step 5: moveit_task → 水平偏移位 (x + x_distance)                  │
│         [cartesian=True, 笛卡尔直线]                                │
│         抬升后水平移开,避免碰撞                                      │
└─────────────────────────────────────────────────────────────────────┘
完整 place 流程(同结构,反向操作)

与 pick 相同的运动轨迹,但 resource_manager 调用变为 resource_manager(resource, "world")——在 Step 3 处将资源从末端执行器释放到世界坐标系。


2.1.10 MoveIt2 API 调用汇总

MoveitInterface 使用的 MoveIt2 接口及其调用位置:

MoveIt2 方法 调用位置 用途
MoveIt2(...) 构造 post_init L51-60 为每个 MoveGroup 创建实例
.allowed_planning_time = 3.0 post_init L61 设置规划超时时间
.max_velocity = speed_ moveit_task L119, moveit_joint_task L151 动态设置最大速度缩放因子
.max_acceleration = speed_ moveit_task L120, moveit_joint_task L152 动态设置最大加速度缩放因子
.move_to_pose(...) moveit_task L129-137 笛卡尔空间运动规划与执行
.wait_until_executed() moveit_task L138, moveit_joint_task L157 阻塞等待运动完成
.move_to_configuration(...) moveit_joint_task L156 关节空间运动规划与执行
.compute_fk(...) pick_and_place L244, moveit_joint_task L160 正运动学:关节角 → 末端位姿
.compute_ik(...) pick_and_place L298-300 逆运动学:末端位姿 → 关节角(含约束)
.end_effector_name pick_and_place L218 获取末端执行器 link 名
.joint_names pick_and_place L232, L308, L313 获取关节名列表

2.1.11 错误处理策略

场景 处理方式
FK 计算失败 最多重试 retry 次(每次间隔 0.1s),超时返回 result.success = False
IK 计算失败 同上
运动规划失败 moveit_task / moveit_joint_task 中最多重试 retry+1
动作序列中任一步失败 pick_and_place 立即中止并返回 result.success = False
未知异常 pick_and_placeset_status 捕获 Exception重置 cartesian_flag,返回失败

2.1.12 cartesian_flag 状态机

cartesian_flag 控制 moveit_task 中是否使用笛卡尔直线规划。在 pick_and_place 执行过程中,它被动态切换:

执行前: cartesian_flag = (上次残留状态)

动作序列执行:
    Step 0: cartesian_flag ← False   (自由空间规划,适合大范围移动)
    Step 1: cartesian_flag ← True    (笛卡尔直线,适合精确操作)
    Step 2: cartesian_flag ← True
    ...
    Step N: cartesian_flag ← True

异常时: cartesian_flag ← False      (重置,防止残留影响后续操作)

这种设计的考量:第一步通常是从安全位移动到工作区附近(距离远、可能需要绕障),使用自由空间规划更灵活;后续步骤是在工作区内的精确操作(下降、抬升、平移),笛卡尔直线规划确保路径可预测。


2.1.13 数据流总览

外部系统 (base_device_node)
    │
    │  JSON 指令字符串
    ▼
┌── MoveitInterface ──────────────────────────────────────────────────┐
│                                                                     │
│  set_position(cmd) ──→ moveit_task() ──→ MoveIt2.move_to_pose()    │
│                                                                     │
│  set_status(cmd) ──→ moveit_joint_task() ──→ MoveIt2.move_to_config│
│                                                                     │
│  pick_and_place(cmd)                                                │
│    │                                                                │
│    ├─ MoveIt2.compute_fk() ─── /compute_fk service ──→ move_group  │
│    ├─ MoveIt2.compute_ik() ─── /compute_ik service ──→ move_group  │
│    ├─ moveit_task()        ─── /move_action         ──→ move_group  │
│    ├─ moveit_joint_task()  ─── /move_action         ──→ move_group  │
│    └─ resource_manager()   ─── SendCmd Action       ──→ tf_update   │
│                                                                     │
│  内部状态:                                                           │
│    joint_poses  ← 预定义位姿查找表                                    │
│    moveit2{}    ← MoveIt2 实例池 (per MoveGroup)                     │
│    tf_buffer    ← TF2 坐标变换缓存                                    │
│    cartesian_flag ← 规划模式状态机                                    │
│                                                                     │
└─────────────────────────────────────────────────────────────────────┘

3. resource_visalization.py — 场景构建与 MoveIt2 启动

路径: unilabos/device_mesh/resource_visalization.py
行数: 429 行
角色: 根据实验室设备/资源配置,动态生成 URDF/SRDF并启动 MoveIt2 所需的全部 ROS 2 节点。

3.1 核心类:ResourceVisualization

3.1.1 模型加载与 URDF 生成

初始化阶段完成以下工作:

  1. 遍历设备注册表:区分 resource(静态资源如 plate/deckdevice(可动设备如机械臂)
  2. Resource 类型:记录 mesh 路径和 TF 变换信息,后续用于碰撞场景
  3. Device 类型:通过 xacro 宏注入动态参数:
    • 设备名前缀(device_name
    • 位置/旋转(从配置中提取)
    • 自定义设备参数(device_config
  4. MoveIt 设备class 包含 moveit.):额外加载:
    • ros2_control xacro 宏 → 生成 ros2_control 硬件接口描述
    • SRDF xacro 宏 → 生成运动学语义描述move group 定义、自碰撞矩阵等)

3.1.2 MoveIt2 配置初始化 (moveit_init)

对每个 MoveIt 设备,加载并合并以下配置文件(均加上设备名前缀):

配置文件 生成目标 作用
ros2_controllers.yaml ros2_controllers_yaml 定义 ros2_control 控制器JointTrajectoryController 等)
moveit_controllers.yaml moveit_controllers_yaml 定义 MoveIt 控制器映射
kinematics.yaml moveit_nodes_kinematics 定义运动学求解器参数

所有关节名和控制器名都加上设备 ID 前缀,确保多设备共存时不冲突。

3.1.3 ROS 2 Launch 节点启动

create_launch_description() 方法启动以下 ROS 2 节点:

节点 作用
ros2_control_node controller_manager ros2_control 硬件管理器,加载 URDF 和控制器配置
spawner (per controller) controller_manager 激活各 JointTrajectoryController
spawner (joint_state_broadcaster) controller_manager 广播关节状态到 /joint_states
robot_state_publisher robot_state_publisher 根据 URDF 和关节状态发布 TF
move_group moveit_ros_move_group MoveIt2 核心节点,提供运动规划服务
rviz2 (可选) rviz2 3D 可视化

move_group 节点的参数配置包括:

  • robot_description: 动态生成的 URDF
  • robot_description_semantic: 动态生成的 SRDF
  • robot_description_kinematics: 合并后的运动学配置
  • planning_pipelines: 从 moveit_configs_utils 加载的规划器配置(默认 OMPL
  • moveit_controllers_yaml: 控制器映射配置
  • robot_description_planning: 速度/加速度限制

4. 三个文件的协作关系

4.1 启动阶段

resource_visalization.py
    │
    ├── 1. 解析设备/资源配置 → 生成 URDF + SRDF
    ├── 2. 合并控制器/运动学配置(带设备名前缀)
    ├── 3. 启动 ros2_control_node + controller spawners
    ├── 4. 启动 robot_state_publisher发布 TF
    ├── 5. 启动 move_group提供规划服务
    └── 6. (可选) 启动 rviz2

4.2 运行阶段

外部调用(如 base_device_node
    │
    ▼
moveit_interface.py
    │
    ├── pick_and_place() ──┬── compute_fk()    ──→ moveit2.py → /compute_fk service
    │                      ├── compute_ik()    ──→ moveit2.py → /compute_ik service
    │                      ├── move_to_pose()  ──→ moveit2.py → /move_action or /plan + /execute
    │                      ├── move_to_config()──→ moveit2.py → /move_action or /plan + /execute
    │                      └── resource_manager()→ SendCmd Action → tf_update
    │
    ├── set_position() ────→ moveit_task()     ──→ moveit2.py → move_to_pose()
    │
    └── set_status() ──────→ moveit_joint_task()→ moveit2.py → move_to_configuration()

4.3 MoveIt2 消息类型使用一览

消息/服务/Action 文件 用途
moveit_msgs/action/MoveGroup moveit2.py 一体化规划+执行
moveit_msgs/action/ExecuteTrajectory moveit2.py 独立轨迹执行
moveit_msgs/srv/GetMotionPlan moveit2.py 运动规划
moveit_msgs/srv/GetCartesianPath moveit2.py 笛卡尔路径规划
moveit_msgs/srv/GetPositionFK moveit2.py 正运动学
moveit_msgs/srv/GetPositionIK moveit2.py 逆运动学
moveit_msgs/srv/GetPlanningScene moveit2.py 获取规划场景
moveit_msgs/srv/ApplyPlanningScene moveit2.py 应用规划场景
moveit_msgs/msg/CollisionObject moveit2.py 碰撞物体管理
moveit_msgs/msg/AttachedCollisionObject moveit2.py 附着碰撞物体
moveit_msgs/msg/Constraints moveit2.py, moveit_interface.py 目标/路径约束
moveit_msgs/msg/JointConstraint moveit2.py, moveit_interface.py 关节约束
moveit_msgs/msg/PlanningScene moveit2.py 规划场景

5. 注册表中的 model 字段与 3D 模型加载

5.1 model 字段概述

在 Uni-Lab-OS 的设备/资源注册表 YAML 中,model 字段是连接注册表定义3D 可视化系统的关键。它告诉 ResourceVisualization 如何为该设备或资源加载 3D 模型。

arm_slider 为例:

# unilabos/registry/devices/robot_arm.yaml (L355-358)
model:
  mesh: arm_slider
  path: https://uni-lab.oss-cn-zhangjiakou.aliyuncs.com/uni-lab/devices/arm_slider/macro_device.xacro
  type: device

三个字段的作用:

字段 说明
mesh arm_slider 模型文件夹名,对应 device_mesh/devices/arm_slider/ 目录
path https://...macro_device.xacro 模型文件的远程下载地址OSS用于首次部署时下载模型资源
type device 模型类型标识,决定 ResourceVisualization 的处理逻辑

5.2 type 字段的两种取值

model.type 决定了 ResourceVisualization 如何加载和处理 3D 模型,有两条完全不同的路径:

type: device — 动态设备(如机械臂、龙门架)

# resource_visalization.py L121-163
if model_config['type'] == 'device':
    # 1. 通过 xacro include 加载设备的 URDF 宏
    #    → device_mesh/devices/{mesh}/macro_device.xacro
    
    # 2. 调用 xacro 宏,注入参数:
    #    parent_link, mesh_path, device_name, x/y/z, rx/ry/r, device_config...
    
    # 3. 若设备 class 包含 "moveit."
    #    → 额外加载 ros2_control xacro 和 SRDF xacro
    #    → 注册为 moveit_nodes后续由 moveit_init() 加载控制器配置

处理流程:

注册表 model.mesh = "arm_slider"
    │
    ├── 加载 URDF:  device_mesh/devices/arm_slider/macro_device.xacro
    │   → 生成完整的 link/joint 运动链,嵌入到全局 URDF 中
    │
    ├── 注入位置参数:  x, y, z, rx, ry, r (从节点配置 position/rotation 中读取, 单位 mm→m)
    │
    ├── 注入设备参数:  device_config 中的键值对作为 xacro 参数
    │
    └── 若 class 含 "moveit.":
        ├── 加载 ros2_control:  config/macro.ros2_control.xacro
        ├── 加载 SRDF:          config/macro.srdf.xacro
        └── 记录到 moveit_nodes → moveit_init() 加载 controllers/kinematics

type: resource — 静态资源(如微孔板、试管架)

# resource_visalization.py L111-120
if model_config['type'] == 'resource':
    # 只记录 mesh 文件路径和 TF 偏移,用于碰撞场景
    # 不注入到 URDF 运动链中
    resource_model[node_id] = {
        'mesh': f"device_mesh/resources/{mesh}",
        'mesh_tf': model_config['mesh_tf']
    }

Resource 类型的 model 结构更丰富,包含 TF 偏移和子物体:

# 例: registry/resources/opentrons/plates.yaml
model:
  mesh: tecan_nested_tip_rack/meshes/plate.stl      # 主体 meshSTL 文件)
  mesh_tf: [0.064, 0.043, 0, -1.5708]               # 位姿偏移 [x, y, z, rotation]
  children_mesh: generic_labware_tube_10_75/meshes/0_base.stl   # 子物体 mesh
  children_mesh_tf: [0.0018, 0.0018, 0, -1.5708]               # 子物体偏移

5.3 两种类型的对比

对比项 type: device type: resource
模型格式 xacro 宏(动态参数化 URDF STL 静态 mesh 文件
加载方式 xacro include → 嵌入全局 URDF 记录路径 → 后续作为碰撞体添加
位置来源 节点配置中的 position/rotation mesh_tf 偏移数组
是否有关节 prismatic/revolute 否(纯静态)
支持 MoveIt 是(通过 class 名中的 moveit. 触发)
子物体 无(运动链本身定义了所有部件) 可选 children_mesh(如管架中的管子)
远程路径 path 字段指向 OSS 下载地址 类似,children_mesh_path 指向子物体
存放目录 device_mesh/devices/{mesh}/ device_mesh/resources/{mesh}
实际示例 arm_slider, toyo_xyz, elite_robot 微孔板, tip rack, 试管架

5.4 arm_slider 注册表完整结构

arm_slider 的注册表键名 robotic_arm.SCARA_with_slider.moveit.virtual 本身就编码了重要信息:

robotic_arm           → 设备大类(机械臂)
  .SCARA_with_slider  → 具体型号SCARA 构型 + 线性滑轨)
  .moveit             → ★ 标记为 MoveIt 设备class 名包含 "moveit."
  .virtual            → 仿真/虚拟设备

class 名中包含 moveit. 的关键作用ResourceVisualization 在 L151 通过 node['class'].find('moveit.') != -1 判断是否需要加载 MoveIt 配置。这是 MoveIt 设备与普通 device 的唯一判别条件——即使两者的 model.type 都是 device

注册表中的关键部分:

robotic_arm.SCARA_with_slider.moveit.virtual:
  # 设备驱动类
  class:
    module: unilabos.devices.ros_dev.moveit_interface:MoveitInterface
    type: python
    action_value_mappings:
      pick_and_place: ...    # SendCmd ActionJSON 指令)
      set_position: ...      # SendCmd Action
      set_status: ...        # SendCmd Action
      auto-moveit_task: ...  # 自动发现的方法UniLabJsonCommand
      auto-moveit_joint_task: ...
      auto-resource_manager: ...
      auto-post_init: ...
  
  # 初始化参数
  init_param_schema:
    config:
      properties:
        moveit_type: ...     # → 对应 device_mesh/devices/{moveit_type}/ 文件夹
        joint_poses: ...     # → 预定义关节位姿查找表
        rotation: ...
        device_config: ...
  
  # ★ 3D 模型定义
  model:
    mesh: arm_slider         # → device_mesh/devices/arm_slider/
    path: https://...        # → OSS 远程下载地址
    type: device             # → ResourceVisualization 按 device 逻辑加载

5.5 从注册表到 MoveIt2 的完整链路

注册表 YAML
│
│  model.mesh = "arm_slider"
│  model.type = "device"
│  class = "robotic_arm.SCARA_with_slider.moveit.virtual"
│                                         ^^^^^^
│                                    class 含 "moveit."
▼
ResourceVisualization.__init__()
│
├── model.type == "device"
│   └── xacro include: devices/arm_slider/macro_device.xacro → URDF
│
├── class 含 "moveit."
│   ├── xacro include: devices/arm_slider/config/macro.ros2_control.xacro → URDF
│   ├── xacro include: devices/arm_slider/config/macro.srdf.xacro → SRDF
│   └── moveit_nodes["device_id"] = "arm_slider"
│
▼
ResourceVisualization.moveit_init()
│
├── 加载 devices/arm_slider/config/ros2_controllers.yaml
├── 加载 devices/arm_slider/config/moveit_controllers.yaml
├── 加载 devices/arm_slider/config/kinematics.yaml
└── 合并到全局配置(带设备名前缀)
│
▼
ResourceVisualization.create_launch_description()
│
├── 启动 ros2_control_node加载 URDF + 控制器配置)
├── 启动 controller spawners激活 arm_controller、gripper_controller
├── 启动 robot_state_publisher发布 TF
├── 启动 move_groupMoveIt2 核心,加载 SRDF + kinematics + planners
└── (可选) 启动 rviz2
│
▼
MoveitInterface.post_init()
│
├── 读取 devices/arm_slider/config/move_group.json
├── 为 "arm" 组创建 MoveIt2 实例
└── 等待 tf_update Action Server 就绪
│
▼
运行时: pick_and_place / set_position / set_status

6. 设备模型文件夹结构MoveIt 设备 vs 非 MoveIt 设备

device_mesh/devices/ 下的每个子文件夹代表一种设备类型的 3D 模型和配置。根据设备是否需要 MoveIt2 运动规划,文件夹内容有显著差异

5.1 非 MoveIt 设备示例:slide_w140 / hplc_station

非 MoveIt 设备只需要 3D 可视化和简单的关节控制(由 liquid_handler_joint_publisher 等自定义节点直接控制),不需要运动规划

slide_w140/                        hplc_station/
├── macro_device.xacro             ├── macro_device.xacro
├── joint_config.json              ├── joint_config.json
├── param_config.json              ├── param_config.json
└── meshes/                        └── meshes/
    └── *.STL                          └── *.STL

仅 3 个配置文件:

文件 作用
macro_device.xacro URDF 模型xacro 宏),定义 link/joint/visual/collision
joint_config.json 关节名与轴向信息,供自定义关节发布器使用
param_config.json 设备尺寸等可配参数(如轨道长度),注入到 xacro 宏参数中

特点:

  • 没有 config/ 子文件夹
  • 没有 SRDF、ros2_control、MoveIt 控制器等配置
  • 关节由应用层直接发布 JointState,不经过 ros2_control 和 MoveIt2

5.2 MoveIt 设备示例:arm_slider

MoveIt 设备需要完整的运动规划支持——从 ros2_control 硬件抽象到 MoveIt2 运动学求解和碰撞矩阵。

arm_slider/
├── macro_device.xacro                 ← URDF 模型xacro 宏)
├── joint_limit.yaml                   ← 关节物理限制effort/velocity/position
├── meshes/                            ← 3D 网格文件
│   ├── arm_slideway.STL
│   ├── arm_base.STL
│   ├── arm_link_1.STL
│   ├── arm_link_2.STL
│   ├── arm_link_3.STL
│   ├── gripper_base.STL
│   ├── gripper_right.STL
│   └── gripper_left.STL
│
└── config/                            ← ★ MoveIt 设备独有的配置目录
    ├── macro.ros2_control.xacro       ← ros2_control 硬件接口定义
    ├── macro.srdf.xacro               ← SRDF 语义描述Move Group + 碰撞矩阵)
    ├── move_group.json                ← Move Group 定义(供 MoveitInterface 使用)
    ├── ros2_controllers.yaml          ← ros2_control 控制器配置
    ├── moveit_controllers.yaml        ← MoveIt ↔ 控制器映射
    ├── kinematics.yaml                ← 运动学求解器配置
    ├── joint_limits.yaml              ← MoveIt 用关节限制(速度/加速度缩放)
    ├── initial_positions.yaml         ← 仿真初始关节位置
    ├── pilz_cartesian_limits.yaml     ← Pilz 笛卡尔限制
    └── moveit_planners.yaml           ← 规划器配置

比非 MoveIt 设备多出 10 个配置文件,全部位于 config/ 子目录。


5.3 arm_slider 各文件详解

5.3.1 macro_device.xacro — URDF 运动链定义

定义了 arm_slider 的完整运动链,包含 8 个 link 和 7 个 joint

world
  └── [fixed] base_link_joint
      └── device_link
          └── [fixed] device_link_joint
              └── arm_slideway (底座滑轨, 有 visual + collision mesh)
                  └── [prismatic, X轴] arm_base_joint (滑轨平移)
                      └── arm_base (机械臂底座)
                          └── [prismatic, Z轴] arm_link_1_joint (升降)
                              └── arm_link_1
                                  └── [revolute, Z轴] arm_link_2_joint (旋转关节1)
                                      └── arm_link_2
                                          └── [revolute, Z轴] arm_link_3_joint (旋转关节2)
                                              └── arm_link_3
                                                  └── [revolute, Z轴] gripper_base_joint (夹爪旋转)
                                                      └── gripper_base (夹爪底座)
                                                          ├── [prismatic, X轴] gripper_right_joint
                                                          │   └── gripper_right
                                                          └── [prismatic, X轴, mimic] gripper_left_joint
                                                              └── gripper_left

关键设计点:

  • 混合关节类型:包含 prismatic滑轨平移 + 升降 + 夹爪)和 revolute旋转关节
  • Mimic 关节gripper_left_joint 通过 <mimic> 标签跟随 gripper_right_joint,实现对称夹爪联动
  • 参数化前缀:所有 link/joint 名都带 ${station_name}${device_name} 前缀,支持多实例
  • 外部关节限制:从 joint_limit.yaml 加载 effort/velocity/position 范围
  • 完整物理属性:每个 link 都有 <inertial>(质量、惯量矩阵)、<visual>STL mesh<collision>(碰撞体)

5.3.2 joint_limit.yaml — 关节物理限制

定义每个关节的运动范围和动力学参数,被 macro_device.xacro 引用:

关节 类型 范围 说明
arm_base_joint prismatic 0 ~ 1.5m 滑轨水平行程
arm_link_1_joint prismatic 0 ~ 0.6m 升降行程
arm_link_2_joint revolute -95° ~ 95° 第一旋转关节
arm_link_3_joint revolute -195° ~ 195° 第二旋转关节
gripper_base_joint revolute -95° ~ 95° 夹爪旋转
gripper_right/left_joint prismatic 0 ~ 0.03m 夹爪开合

5.3.3 config/macro.ros2_control.xacro — ros2_control 硬件接口

定义 ros2_control 硬件抽象层,将关节映射到控制接口:

  • 硬件插件mock_components/GenericSystem(仿真模式,可替换为真实硬件驱动)
  • 每个关节声明
    • command_interface: position — 位置控制模式
    • state_interface: position — 位置反馈(含 initial_valueinitial_positions.yaml 加载)
    • state_interface: velocity — 速度反馈
  • 6 个关节arm_base_joint ~ gripper_right_jointgripper_left_joint 因为是 mimic 关节,不需要独立控制接口)

5.3.4 config/macro.srdf.xacro — SRDF 语义描述

MoveIt2 的语义机器人描述,定义了:

Move Groups规划组

组名 类型 内容
{device_name}arm chain arm_slidewaygripper_base5 DOF 运动链)
{device_name}arm_gripper joint gripper_right_joint(夹爪控制)

Disable Collisions自碰撞矩阵

22 条 <disable_collisions> 规则,标记不可能碰撞的 link 对Adjacent / Never减少碰撞检测计算量。例如

  • arm_basearm_slidewayAdjacent相邻 link必然接触
  • arm_link_1arm_link_3Never物理上不可能碰撞
  • gripper_leftgripper_rightNever

5.3.5 config/move_group.json — MoveitInterface 配置

MoveitInterface.post_init() 使用,定义每个 Move Group 的关节和端点:

{
    "arm": {
        "joint_names": ["arm_base_joint", "arm_link_1_joint",
                        "arm_link_2_joint", "arm_link_3_joint",
                        "gripper_base_joint"],
        "base_link_name": "device_link",
        "end_effector_name": "gripper_base"
    }
}

MoveitInterface 读取此文件后,为 "arm" 组创建一个 MoveIt2 实例,自动加上设备名前缀。

5.3.6 config/ros2_controllers.yaml — 控制器定义

定义两个 JointTrajectoryController

控制器 控制的关节 说明
arm_controller arm_base_joint ~ gripper_base_joint (5个) 机械臂主体
gripper_controller gripper_right_joint (1个) 夹爪

resource_visalization.pymoveit_init() 读取,加上设备名前缀后合并到全局 ros2_controllers_yaml 中。

5.3.7 config/moveit_controllers.yaml — MoveIt ↔ 控制器映射

告诉 MoveIt2 的 move_group 节点如何将规划好的轨迹发送到 ros2_control 控制器:

  • arm_controller → FollowJointTrajectory Action5 个关节)
  • gripper_controller → FollowJointTrajectory Action1 个关节)

5.3.8 config/kinematics.yaml — 运动学求解器

arm:
  kinematics_solver: lma_kinematics_plugin/LMAKinematicsPlugin
  kinematics_solver_search_resolution: 0.005
  kinematics_solver_timeout: 0.005

使用 LMA (Levenberg-Marquardt Algorithm) 运动学求解器进行正/逆运动学计算。这是 MoveIt2 的通用 IK 求解器,适用于任意运动链拓扑。

5.3.9 其他配置文件

文件 作用
initial_positions.yaml 仿真启动时各关节初始角度/位置(所有为 0夹爪张开 0.03m
joint_limits.yaml MoveIt 层面的速度/加速度缩放限制(覆盖 URDF 中的值)
pilz_cartesian_limits.yaml Pilz 工业运动规划器的笛卡尔速度/加速度限制
moveit_planners.yaml 可用规划器列表(ompl_interface/OMPLPlanner

5.4 对比:toyo_xyz(另一个 MoveIt 设备)

toyo_xyz 是一个三轴直线运动平台XYZ 龙门),也是 MoveIt 设备。与 arm_slider 对比:

对比项 arm_slider toyo_xyz
自由度 5 DOF (2 prismatic + 3 revolute) + 夹爪 3 DOF (3 prismatic)
运动链 混合链(平移+旋转) 纯直线链(全 prismatic
Move Group arm (5 joints) + arm_gripper (1 joint) toyo_xyz (3 joints)
末端执行器 gripper_base slider3_link
独有文件 joint_limit.yaml joint_config.json + param_config.json
xacro 参数 固定尺寸 可配长度 (length1/2/3) + mesh scale 缩放
IK 求解器 LMA LMA

toyo_xyz 的额外特点:

  • param_config.json:定义三轴行程和滑块尺寸,通过 xacro 参数动态缩放 STL 模型
  • joint_config.json:简单的关节名→轴向映射,供非 MoveIt 的关节发布器使用
  • config/full_dev.urdf.xacro:额外的完整 URDF 文件(独立调试用)

两者的 MoveIt 配置文件结构完全一致SRDF、ros2_control、controllers、kinematics说明 Uni-Lab-OS 的 MoveIt 设备遵循统一的模板。


5.5 MoveIt vs 非 MoveIt 设备文件对比总结

非 MoveIt 设备 (slide_w140)           MoveIt 设备 (arm_slider)
───────────────────────────           ───────────────────────────
macro_device.xacro          ✓         macro_device.xacro           ✓
joint_config.json           ✓         joint_limit.yaml             ✓
param_config.json           ✓         
                                      config/
                                      ├── macro.ros2_control.xacro ★ ros2_control 硬件接口
                                      ├── macro.srdf.xacro         ★ SRDF (Move Group + 碰撞)
                                      ├── move_group.json          ★ MoveitInterface 配置
                                      ├── ros2_controllers.yaml    ★ 控制器定义
                                      ├── moveit_controllers.yaml  ★ MoveIt↔控制器映射
                                      ├── kinematics.yaml          ★ IK 求解器配置
                                      ├── joint_limits.yaml        ★ MoveIt 关节限制
                                      ├── initial_positions.yaml   ★ 仿真初始状态
                                      ├── pilz_cartesian_limits.yaml ★ 笛卡尔限制
                                      └── moveit_planners.yaml     ★ 规划器列表

文件数: 3                             文件数: 12 (3 + 10 MoveIt 专用)

核心区别MoveIt 设备多出的 config/ 目录下的 10 个文件,构成了 MoveIt2 运动规划所需的完整配置栈:

  1. 硬件层 (macro.ros2_control.xacro): 关节如何被控制
  2. 语义层 (macro.srdf.xacro): 哪些关节组成规划组,哪些碰撞可以忽略
  3. 规划层 (kinematics.yaml, moveit_planners.yaml, pilz_cartesian_limits.yaml): 如何求解和规划
  4. 执行层 (ros2_controllers.yaml, moveit_controllers.yaml): 轨迹如何下发到控制器
  5. 桥接层 (move_group.json): Uni-Lab-OS 的 MoveitInterface 如何连接到 MoveIt2

6. 设计特点

  1. 多设备支持:通过设备名前缀机制(device_id_所有关节名、link 名、控制器名都是唯一的,支持在同一 ROS 2 环境中运行多台机械臂。

  2. 动态场景构建ResourceVisualization 根据实验室配置动态生成 URDF/SRDF无需手动编写或维护静态模型文件。

  3. 规划/执行分离MoveIt2 类支持 MoveGroup Action合并模式和 Plan+Execute分离模式可根据场景灵活选择。

  4. 线程安全MoveIt2 类通过 threading.Lock 保护关节状态和执行状态的并发访问。

  5. 碰撞场景集成:支持完整的碰撞物体生命周期管理(添加/移动/附着/分离/删除),可在运行时动态更新规划场景。

  6. 资源 TF 动态更新MoveitInterface 通过 resource_manager() 在 pick/place 时动态更新资源的 TF 父 link实现物体在机器人和环境之间的"跟随"效果。

  7. 统一设备模板MoveIt 设备遵循统一的 config/ 目录结构SRDF、ros2_control、controllers、kinematics新增设备只需按模板创建配置文件即可接入 MoveIt2 运动规划。