系统Ros1到Ros2是一个迭代过程,ROS2做为全新得技术非常值得学习和领悟,本文介绍了统一这些构建工具以及扩展应用领域的步骤,以及设计原则。
在 ROS 生态系统中,软件被分成许多包。开发人员同时处理多个包是很常见的。这与开发人员一次只在一个软件包上工作的工作流程形成对比,并且所有依赖项都提供一次但不会被迭代。
构建一组包的“手动”方法包括:按拓扑顺序一个接一个地构建所有包。对于每个包,文档通常会描述依赖项是什么,如何设置环境来构建包以及如何设置环境以使用包。如果无将该过程自动化的工具,这样的工作流程在规模上是不切实际的。
构建工具执行通过一次调用构建一组包的任务。对于 ROS 1,有多种不同的工具对此提供支持,即 catkin_make、catkin_make_isolated 和 catkin_tools。对于 ROS 2 到 Ardent 版本,提供此功能的构建工具称为 ament_tools。
统一构建工具的目标是通过一次调用构建一组包。它应该与 ROS 1 包以及在其清单文件中提供必要信息的 ROS 2 包一起使用。它还应该与本身不提供清单文件的包一起使用,因为可以推断和/或从外部提供必要的元信息。这将允许构建工具用于非 ROS 包(例如 Gazebo,包括其启动依赖项、sdformat 等)。
在 ROS 生态系统中,已经存在一些支持此用例的工具(见下文)。每个现有工具都执行类似的任务并复制大量逻辑。由于是单独开发的,某些功能仅在某些工具中可用,而其他工具则缺乏这些功能。
使用单一通用构建工具的原因归结为减少开发和维护所需的工作量。此外,这使得新开发的功能可用于所有用例。
由于本文关注的是构建工具,因此需要澄清与构建系统的区别。
构建工具对一组包进行操作。它确定依赖关系图并按拓扑顺序为每个包调用特定的构建系统。构建工具本身应该尽可能少地了解用于特定包的构建系统。足以知道如何为其设置环境、调用构建并设置环境以使用构建的包。现有的 ROS 构建工具有:catkin_make、catkin_make_isolated、catkin_tools 和 ament_tools。
另一方面,构建系统在单个包上运行。示例包括 Make、CMake、Python setuptools 或 Autotools(未在 ROS atm 中使用)。 CMake 包是例如通过调用以下步骤进行构建:cmake、make、make install。
catkin 和 ament_cmake 都基于 CMake,并提供如下所述的一些便利功能。
除了包的实际构建之外,一个非常重要的部分是环境设置。例如,为了让 CMake 项目使用 CMake 函数 find_package 发现依赖项,该依赖项的 CMake 模块(例如 FindFoo.cmake)或 CMake 配置文件(例如 FooConfig.cmake)必须位于 CMake 的前缀中隐式搜索(例如 /usr)或必须通过环境变量 CMAKE_PREFIX_PATH / CMAKE_MODULE_PATH 提供位置。
除了在另一个包之上构建一个包(在 CMake 的情况下使用 find_package),您可能需要调整环境以便从包中运行可执行文件。例如,当包在非默认位置安装共享库时,需要扩展环境变量 LD_LIBRARY_PATH(或 Windows 上的 PATH)以包含包含文件夹,然后再尝试运行在运行时加载该库的可执行文件。
设置这些环境变量的功能可以由构建工具或构建系统提供。在后一种情况下,构建工具只需要知道构建系统如何公开环境设置以便重用它。
考虑到用户可能想要手动调用每个包的构建系统的用例,如果构建系统已经提供了尽可能多的环境设置,那将是有益的。这样可以避免在不使用构建工具时强制用户手动处理环境设置。
为了阐明本文的范围,我们明确列举了一些相关主题,即使它们没有被考虑。
本文不考虑任何与构建系统相关的功能(与构建工具没有直接关系)。
统一的构建工具将支持不同的构建系统以满足所描述的目标。使用不同构建系统的包是否正确地相互操作也很大程度上取决于构建系统。虽然构建工具应确保它不会阻止该用例,但本文不会涵盖在单个工作区中混合多个构建系统的用例(例如,使用 catkin 的 ROS 1 包和使用 ament_cmake 的 ROS 2 包)。
构建工具不包括获取要构建的包的源代码所必需的步骤。已经有工具可以帮助解决这个问题。例如,需要获取的存储库列表由手工制作的 .rosinstall 或 .repos 文件提供,或者使用 rosinstall_generator 生成一个。然后可以使用多种工具之一获取存储库列表,例如 rosinstall 或 wstool 用于 .rosinstall 文件,或 vcstool 用于 .repos 文件。
构建工具也不提供安装构建包所需的任何依赖项的机制。在 ROS 生态系统中,rosdep 可以用于此。
构建工具也不会创建二进制包(例如 Debian 包)。在 ROS 生态系统中,bloom 用于生成所需的元数据,然后依赖平台的工具(如 dpkg-buildpackage)构建二进制包。
下面简要介绍 ROS 生态系统中使用的构建系统。
CMake 是一个跨平台的构建系统生成器。项目使用独立于平台的 CMakeLists.txt 文件指定其构建过程。用户通过使用 CMake 为他们平台上的原生工具生成构建系统来构建项目,例如Makefile 或 Visual Studio 项目。
catkin 基于 CMake 并提供了一组方便的功能,使编写 CMake 包更容易。它自动生成 CMake 配置文件以及 pkg-config 文件。它还提供了注册不同类型测试的功能。
使用 catkin 的包在名为 package.xml 的清单文件中指定其元数据。清单文件的最新格式在 ROS REP 149 中指定。
ament_cmake 是 catkin 的演变,也是基于 CMake 的。 ament_cmake 和 catkin 的主要区别在另一篇文章中描述。在构建工具的上下文中,最大的区别是 ament_cmake 生成特定于包的文件来设置环境以在构建和安装包后使用它。
使用 ament_cmake 的包使用与 catkin 相同的清单文件(除了它需要格式版本 2 或更高版本)。
setuptools 是打包 Python 包的常用工具。 Python 包使用 setup.py 文件来描述依赖关系以及构建和安装的方式和内容。在 ROS 2 中,包可以是“vanilla” Python 包,而在 ROS 1 中,任何 Python 功能都是从 CMake 文件触发的。
ROS 生态系统中已经使用了几种不同的构建工具。以下小节将描述它们的操作方法以及它们的优点和缺点。
catkin_make 由 ROS 包 catkin 提供,其中包含 ROS 1 的构建系统。它被设计为 ROS Fuerte 的 rosbuild 的继任者。
该工具仅调用一次 CMake,并使用 CMake 的 add_subdirectory 函数在单个上下文中处理所有包。虽然这是一种有效的方法,因为所有包中的所有目标都可以并行化,但它具有明显的缺点。由于单一上下文,所有函数名称、目标和测试在所有包中共享一个命名空间,并且在更大范围内这很容易导致冲突。单一上下文也会受到包之间的副作用影响,有时需要跨包边界添加额外的目标依赖项。
catkin_make 支持构建以下包:
ROS 1 catkin 包,带有 package.xml 文件。
catkin_make_isolated 由 ROS 包 catkin 提供,其中包含 ROS 1 的构建系统。它是在 catkin_make 之后开发的,用于解决在单个 CMake 上下文中构建多个包所涉及的问题。
该工具仅支持基于 CMake 的包,并使用 CMake 包常见的命令序列按拓扑顺序构建每个包:cmake、make、make install。虽然每个包都可以并行化其目标的构建,但即使它们不是彼此的(递归)依赖关系,它们也会按顺序处理。
catkin_make_isolated 支持构建以下包:
- ROS 1 catkin 包,带有 package.xml 文件。
- 带有 package.xml 文件的普通 CMake 包。
catkin_tools 由用于构建 ROS 1 包的独立 Python 包提供。它是在 catkin_make / catkin_make_isolated 之后开发的,用于并行构建包并提供显着的可用性改进。该工具支持构建 CMake 包并单独构建它们,并支持跨包并行化进程。
catkin_tools 支持构建以下包:
- ROS 1 catkin 包,带有 package.xml 文件。
- 带有 package.xml 文件的普通 CMake 包。
ament_tools
ament_tools 由用于构建 ROS 2 包的独立 Python 3 包提供。它是为引导 ROS 2 项目而开发的,因此仅针对 Python 3,适用于 Linux、MacOS 和 Windows。除了 CMake 包之外,它还支持构建 Python 包,并且可以在不需要显式包清单的情况下推断元信息(例如,用于 FastRTPS 包)。该工具执行类似 catkin_make_isolated 和 catkin_tools 的“隔离”构建(每个包一个 CMake 调用),并且还并行化彼此没有(递归)依赖的包的构建(如 catkin_tools)。虽然它比 catkin_tools 涵盖了更多的构建系统和平台,但它没有 catkin_toolss 的任何可用性功能,如配置文件、输出处理等。
ament_tools 支持构建以下包:
ROS 中现有的构建工具都是由它们所支持的构建系统命名的。一般来说,构建工具应该可以支持多个不同的构建系统。因此,从单个构建系统派生的构建工具的名称可能会误导用户该工具仅适用于该特定构建系统。为避免用户混淆,构建工具应具有不同的不相关名称,以避免暗示不希望的相关性。
统一构建工具应提供现有工具所提供功能的超集。下面描述了几个用例以及所需的软件标准。
其他未明确涵盖但现有工具已支持的用例(例如交叉编译、DESTDIR 支持、构建没有清单的 CMake 包)应继续使用统一构建工具。
统一构建工具应满足以下用例。
该工具需要能够构建已经可以使用 catkin_make_isolated / catkin_tools 构建的 ROS 1 工作区。由实现决定它是仅支持标准 CMake 工作流还是还支持 catkin 的自定义开发空间概念。
在 ROS 2 中,开发空间的概念已被有意删除。将来,在 ROS 1 中提供符号链接安装的概念可能是可行的,以提供类似的好处而没有缺点。
该工具需要能够构建已经可以使用 ament_tools 构建的 ROS 2 工作区。
1)构建包含依赖项的 Gazebo
克隆包含 Gazebo 及其所有依赖项(不包括系统包)的存储库后,该工具需要能够构建包集。可以从外部提供无法从源中推断的元信息,而无需添加或修改工作区中的任何文件。在构建之后,可以获取/调用单个文件来设置环境以使用 Gazebo(例如 GAZEBO_MODEL_PATH)。
2)开发环境设置
为包调用构建系统还意味着在进程之前设置环境变量,例如CMAKE_PREFIX_PATH。开发人员应该可以为一个包手动调用构建系统。环境变量可能与构建后使用包所需的环境变量部分不同。为了方便起见,该工具应提供一种易于使用的机制来设置手动调用构建系统所需的开发环境。
3)超越建筑
构建包只是构建工具可以对一组包执行的一项任务。附加任务,例如运行测试也应该包含在构建工具中。构建工具必须提供这些抽象任务,然后将它们映射到每个支持的构建系统所需的步骤。
4)软件标准
该工具旨在支持各种构建系统、用例和平台。上面提到的主要是由 ROS 生态系统的需求驱动的,但该工具也应该可以在 ROS 生态系统之外使用(例如用于 Gazebo)。因此,它的设计方式应该能够扩展其功能。
假设该工具将在 Python 中实现(因为所有现有的 ROS 构建工具都是这种情况),入口点机制提供了一种使软件可扩展的便捷方法。扩展甚至不必集成到包含构建工具核心逻辑的 Python 包中,但可以很容易地由其他 Python 包提供。这种方法不仅可以促进模块化设计并促进清晰的接口,还可以实现外部贡献,而无需将它们集成到单个整体包中。
5)几个众所周知的软件原则:
6)扩展点
以下项目是提供自定义功能的可能扩展点:
在撰写本文初稿时,现有的构建工具都不支持本文中描述的功能超集。有多种不同的途径可以实现通用构建工具的目标,分为两类:
由于这三个构建工具都没有 catkin_tools 的功能丰富性,因此严格认为开始构建这些构建工具之一的用处不大。因此,这些都不被视为通用构建工具的基础。
由于 catkin_tools 在许多方面都是最完整的 ROS 构建工具,因此它应该是正在进化的工具。虽然 ament_tools 有一些 catkin_tools 当前缺乏的功能(例如,没有清单的普通 CMake 支持、Windows 支持),但 catkin_tools 的丰富功能使其成为更好的起点。
从本文的初稿开始,colcon 项目的开发就考虑到了通用构建工具的目标和要求。在目前的形式下,它已经能够构建 ROS 1 工作区、ROS 2 工作区以及 Gazebo,包括其点火依赖项。它使用 Python 3.5+ 并针对 ROS 支持的所有平台:Linux、macOS 和 Windows。
由于它还没有被很多人使用,因此交叉编译、DESTDIR 等更高级的功能还没有经过测试(因此可能还不能工作)。
对于决策过程,根据上述基本原理,仅考虑以下两个选项:
如果这个话题早点得到解决,一些重复的工作可能就可以避免。一年多前,因为可用资源的限制,开发通用构建工具的工作被暂停。尽管如此,现在就做出决定继续,至少可以避免进一步的不确定性和重复工作。
所考虑的两个选项都具有独特且有价值的功能,并且有很好的论据可以在这两个工具中的任何一个上构建我们的未来发展。由于两者都是用 Python 编写的,这两个工具中的任何一个都可以“转换”以涵盖另一个工具的优点。因此,决定的两个重要标准是:
ROS 2 开发人员目前使用 ROS 2 包构建了一个稳定增长的工作空间。同样的情况也发生在 ci.ros2.org 上的单一 Jenkins 作业中(其优势在于可以轻松地测试跨存储库的更改)。因此,人们急切地期待着能够轻松过滤需要构建的包的功能来改进开发过程。
对于最新的 ROS 2 版本 Ardent,buildfarm build.ros.org 仅提供生成 Debian 软件包的作业。开发作业或拉取请求作业都不可用,也不支持构建本地预发布。对于即将到来的 ROS 2 版本 Bouncy,这些作业类型应该可以用于支持维护者。
在 ROS 2 Bouncy 中,通用构建工具将是推荐的选项。
必要的工作,对于选项 A) 或 B),需要解决以下问题:
需要更新 ci.ros2.org 上的作业和脚本以调用通用构建工具而不是 ament_tools。
需要更新 ros_buildfarm 包以调用通用构建工具而不是 catkin_make_isolated。 ROS 2 buildfarm 将在即将发布的 ROS 2 Bouncy 版本中使用此修改。 ROS 1 buildfarm 将来可以使用相同的修改。
对于选项 A),需要解决以下问题:
对于选项 B),需要解决以下问题:
长期目标是通用构建工具将用于 ROS 1、ROS 2 以及其他非 ROS 项目。目前没有时间线何时该工具将用于 ROS 1 构建或推荐给 ROS 1 用户。这完全取决于 ROS 1 可用的资源。
除此之外,对于这两个选项,还有超越当前目标的后续工作。以下列举了其中的一些,但绝不是详尽的:
1)对于选项 A),应考虑以下项目:
2)对于选项 B),应考虑以下项目:
基于以上信息,决定选择 colcon 作为通用构建工具。
该决定是在考虑了 ROS 2 团队成员和一些 ROS 1 用户的意见后做出的。这个决定并不容易,因为它不是一致的,但绝大多数意见要么是赞成的,要么是矛盾的。
详细说明 colcon 的一个重要优势是它现在可以部署到 ROS 2 并且它涵盖了我们当前的用例。另一个倾向于 colcon 的论点是,在 build.ros2.org 上为即将发布的 Bouncy 版本在所有目标平台上提供开发/公关/预发布工作的预期很少。虽然仍然缺少一些附加功能和可用性选项,但可以在将来有时间和/或需要时添加它们。
catkin_tools 为实现 Bouncy 所描述的目标而进行的必要的前期开发工作将分散 ROS 2 团队的注意力,使其无法将时间花在 ROS 2 本身的功能开发和错误修复上。
虽然短期优势肯定是决定支持 colcon 的主要原因,但它们并不是唯一的。在未来开发此工具时,更简洁的架构、模块化和可扩展性以及 Python 3.5 代码库将是宝贵的长期利益。构建工具名称与支持的构建系统的分离以及与“仅 ROS”工具的分离也有望帮助用户了解差异并吸引新用户和潜在贡献者。
以下后续步骤将在下一个 ROS 2 版本 Bouncy 之前进行。
ROS 2 buildfarm(s) 将更新为使用 colcon 并为 Bouncy 版本提供开发/公关/预发布作业。
ROS 2 wiki 中从源代码构建的说明将更新为使用 colcon。
在 Bouncy 发布之后,ament_tools 存储库将被存档,从 ros2.repos 文件中删除,并且不会发布到未来的 ROS 2 发行版中。
关于客户影响,以下几项简要列举了这一决定对 ROS 开发人员和用户的意义: