# Uni-Lab-OS MoveIt2 集成架构总结 ## 概览 Uni-Lab-OS 通过三个核心文件实现了对 MoveIt2(ROS 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 类属性与实例属性 ```python 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` ```python 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` ```python 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` ```python 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` ```python def set_position(self, command: str) ``` 最简单的运动接口,解析 JSON 指令后直接委托给 `moveit_task`。 **JSON 指令格式:** ```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` ```python def set_status(self, command: str) ``` 将机械臂移动到预定义的关节配置(如 home 位、准备位等),关节角度从 `self.joint_poses` 查找表中获取。 **JSON 指令格式:** ```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` 查找表结构:** ```python { "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` ```python def pick_and_place(self, command: str) ``` 这是 `MoveitInterface` 最复杂的方法,实现了完整的抓取-放置工作流。它动态构建一个**有序函数列表** (`function_list`),然后顺序执行。 **JSON 指令格式(完整参数):** ```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 求解的搜索空间,确保机械臂在抬升/移动过程中保持特定关节(如肘关节)在安全范围内。 ##### 阶段 4A:有 `lift_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 ``` ##### 阶段 4B:无 `lift_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_place` 和 `set_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/deck)和 `device`(可动设备如机械臂) 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` 为例: ```yaml # 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` — 动态设备(如机械臂、龙门架) ```python # 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` — 静态资源(如微孔板、试管架) ```python # 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 偏移和子物体: ```yaml # 例: registry/resources/opentrons/plates.yaml model: mesh: tecan_nested_tip_rack/meshes/plate.stl # 主体 mesh(STL 文件) 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`。 注册表中的关键部分: ```yaml 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 Action(JSON 指令) 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_group(MoveIt2 核心,加载 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` 通过 `` 标签跟随 `gripper_right_joint`,实现对称夹爪联动 - **参数化前缀**:所有 link/joint 名都带 `${station_name}${device_name}` 前缀,支持多实例 - **外部关节限制**:从 `joint_limit.yaml` 加载 effort/velocity/position 范围 - **完整物理属性**:每个 link 都有 ``(质量、惯量矩阵)、``(STL mesh)和 ``(碰撞体) #### 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_value` 从 `initial_positions.yaml` 加载) - `state_interface: velocity` — 速度反馈 - **6 个关节**:`arm_base_joint` ~ `gripper_right_joint`(`gripper_left_joint` 因为是 mimic 关节,不需要独立控制接口) #### 5.3.4 `config/macro.srdf.xacro` — SRDF 语义描述 MoveIt2 的语义机器人描述,定义了: **Move Groups(规划组):** | 组名 | 类型 | 内容 | |------|------|------| | `{device_name}arm` | chain | `arm_slideway` → `gripper_base`(5 DOF 运动链) | | `{device_name}arm_gripper` | joint | `gripper_right_joint`(夹爪控制) | **Disable Collisions(自碰撞矩阵):** 22 条 `` 规则,标记不可能碰撞的 link 对(Adjacent / Never),减少碰撞检测计算量。例如: - `arm_base` ↔ `arm_slideway`:Adjacent(相邻 link,必然接触) - `arm_link_1` ↔ `arm_link_3`:Never(物理上不可能碰撞) - `gripper_left` ↔ `gripper_right`:Never #### 5.3.5 `config/move_group.json` — MoveitInterface 配置 供 `MoveitInterface.post_init()` 使用,定义每个 Move Group 的关节和端点: ```json { "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.py` 的 `moveit_init()` 读取,加上设备名前缀后合并到全局 `ros2_controllers_yaml` 中。 #### 5.3.7 `config/moveit_controllers.yaml` — MoveIt ↔ 控制器映射 告诉 MoveIt2 的 `move_group` 节点如何将规划好的轨迹发送到 ros2_control 控制器: - `arm_controller` → FollowJointTrajectory Action(5 个关节) - `gripper_controller` → FollowJointTrajectory Action(1 个关节) #### 5.3.8 `config/kinematics.yaml` — 运动学求解器 ```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 运动规划。