/nix-ros-workspace

An opinionated builder for ROS workspaces using lopsided98/nix-ros-overlay.

Primary LanguageNix

nix-ros-workspace

An opinionated builder for ROS workspaces using lopsided98/nix-ros-overlay.

Quickstart

To open a shell with ROS 2: Humble Hawksbill, rviz2, and turtlesim:

$ nix-shell \
  --extra-substituters 'https://ros.cachix.org' --extra-trusted-public-keys 'ros.cachix.org-1:dSyZxI8geDCJrwgvCOHDoAfOm5sV1wCPjBkKL+38Rvo=' \
  https://github.com/hacker1024/nix-ros-workspace/archive/master.tar.gz -A cli.env \
  --argstr distro humble \
  --argstr rosPackages 'rviz2 turtlesim'

Or, to build a derivation containing all of the above, use nix-build and remove the .env:

$ nix-build \
  --extra-substituters 'https://ros.cachix.org' --extra-trusted-public-keys 'ros.cachix.org-1:dSyZxI8geDCJrwgvCOHDoAfOm5sV1wCPjBkKL+38Rvo=' \
  https://github.com/hacker1024/nix-ros-workspace/archive/master.tar.gz -A cli \
  --argstr distro humble \
  --argstr rosPackages 'rviz2 turtlesim'

Rationale

lopsided98/nix-ros-overlay provides a variant of buildEnv that allows ROS packages to see each other. This falls short in a few ways, though:

  • ROS 2 is not well supported.
  • Non-ROS packages added to the environment do not get included outside of nix-shell.
  • There is no clear way to set up a development environment with a mix of prebuilt packages and package build inputs.

The buildROSWorkspace function included in this repository aims to solve these issues.

Setup

  1. Set up lopsided98/nix-ros-overlay, ensuring that PR #269 is included.
  2. Add the overlay from this repository ((import /path/to/repository { }).overlay).

Usage

API

buildROSWorkspace is included in the ROS distro package sets. The following examples are designed to be invoked with callPackage, e.g. rosPackages.rolling.callPackage.

buildROSWorkspace takes a derivation name and several sets of packages.

  • devPackages are packages that are under active development. They will be available in the release environment (nix-build), but in the development environment (nix-shell), only the build inputs of the packages will be available.

  • prebuiltPackages are packages that are not under active development (typically third-party packages). They will be available in both the release and development environments.

  • prebuiltShellPackages are packages that will get added only to the development shell environment. This is useful for build tools like GDB.

In order to set a default ROS domain ID, the manualDomainId argument can be set. This defaults to the value of the NRWS_DOMAIN_ID environment variable at evaluation time, or 0 if it is unset.

{ buildROSWorkspace
, rviz2
, my-package-1
, my-package-2
}:

buildROSWorkspace {
  name = "my";
  devPackages = {
    inherit
      my-package-1
      my-package-2;
  };
  prebuiltPackages = {
    inherit
      rviz2;
  };
}

Sibling dependencies

Some packages expect other packages to be available in the workspace, without depending on them directly. Many launch files, for example, attempt to run arbitrary nodes and programs.

To accomodate this, the workspacePackages passthru attribute is available. Packages added to this set will be detected by buildROSWorkspace and added to prebuiltPackages, along with any workspacePackages of their own.

{ buildRosPackage
, xacro
, gazebo-ros
}:

buildRosPackage {
  # ...
  passthru.workspacePackages = {
    inherit
      xacro
      gazebo-ros;
  };
}

Command line

The following examples assume a default.nix exists, evaluating to the result of a buildROSWorkspace call.

Building

To build a workspace as a regular Nix package:

$ nix-build

$ # Then, for example:
$ ./result/bin/ros2 pkg list

To enter a shell in the workspace release environment:

$ nix-shell -p 'import ./. { }'
$ eval "$(mk-workspace-shell-setup)"

$ # Then, for example:
$ ros2 pkg list

Developing

To enter a shell in the workspace development environment:

$ nix-shell -A env

$ # Then, for example:
$ cd ~/ros_ws
$ colcon build
Composition
For

env also includes a "sub-environment" for each package in devPackages. These environments are identical to the main environment, but all packages other than the specified one are moved into prebuiltPackages.

In the example below, my-package-1's build dependencies will be available as normal, but my-package-2 will be available as if it were in prebuiltPackages.

$ nix-shell -A env.for.my-package-1
And

Often, it is useful to work with a subset of the devPackages. This can be done by using the and attributes, which move the selected prebuiltPackages back into the devPackages.

For example, to work with both my-package-1 and my-package-2 as devPackages:

$ nix-shell -A env.for.my-package-1.and.my-package-2

The .for.my-package-1 moves all but my-package-1 into prebuiltPackages, and the .and.my-package-2 brings my-package-2 back.

These techniques are preferable to nix-shell -A my-package-1, as the former will include standard workspace tools and ROS 2 fixes.

Shortcuts

for and and can be left out. These two values are the same:

env.for.my-package-1.and.my-package-2
env.my-package-1.my-package-2