Skip to main content

Compile xacro files to URDF or MJCF from Python or the command line (no ROS required).

Project description

xacrodoc

xacrodoc is a tool for compiling xacro files to plain URDF files or Mujoco MJCF files from within Python code or via the command line. It is fully functional whether ROS is installed on the system or not.

Why?

Compared to the regular xacro package, xacrodoc allows you to:

  • Conveniently convert xacro files to URDF or Mujoco MJCF files without a ROS installation (this includes converting plain URDF to MJCF).
  • Programmatically compose multiple xacro files and apply substitution arguments to build a flexible URDF model directly in your code. This allows you to avoid the clutter of redundant compiled raw URDFs and only keep the xacro source files.
  • Seamlessly obtain URDF strings and (temporary) URDF file paths as needed. For example, many libraries (such as Pinocchio) accept a URDF string to build a model, but others (like PyBullet) only load URDFs directly from file paths.
  • Easily copy assets like meshes to a single directory and have fine-grained control over file protocols and absolute vs. relative paths, if desired.

Documentation

Usage information and examples are provided in this README. The package's full function reference can be found here.

Installation

xacrodoc requires at least Python 3.8. Note that ROS does not need to be installed on the system, but xacrodoc will also use its infrastructure to look for packages if it is available.

The library can be installed from pip:

pip install xacrodoc

# or to also include conversion to MJCF files:
pip install "xacrodoc[mujoco]"

or from source:

git clone --recurse-submodules https://github.com/adamheins/xacrodoc
cd xacrodoc
pip install .

It is recommended to install the command line tool into an isolated environment using uv:

uv tool install xacrodoc

# for conversion to MJCF files, use:
uv tool install "xacrodoc[mujoco]"

or pipx:

pipx install xacrodoc
# or
pipx install "xacrodoc[mujoco]"

Python Usage

Basic

A basic use-case of compiling a URDF from a xacro file:

import os
from xacrodoc import XacroDoc

doc = XacroDoc.from_file("robot.urdf.xacro")

# or relative to a ROS package
# e.g., for a file located at some_ros_package/urdf/robot.urdf.xacro:
doc = XacroDoc.from_package_file("some_ros_package", "urdf/robot.urdf.xacro")

# convert to a string of URDF
urdf_str = doc.to_urdf_string()

# or write to a file
doc.to_urdf_file("robot.urdf")

# or just work with a temp file
# this is useful for working with libraries that expect a URDF *file* (rather
# than a string)
with doc.temp_urdf_file_path() as path:
  # do stuff with URDF file located at `path`...
  # file is cleaned up once context manager is exited

# you can also manage the temp file yourself if you don't want to clean it up
# right away
path = doc.to_temp_urdf_file()
# ...do stuff with path...

# manually delete the temp file
os.remove(path)

Finding ROS packages

xacro files often make use of $(find <pkg>) directives to resolve paths relative to a given ROS package. If ROS is installed on the system, xacrodoc automatically looks for ROS packages using the usual ROS infrastructure. If not, or if you are working with packages outside of a ROS workspace, you'll need to tell xacrodoc where to find packages. There are a few ways to do this:

import xacrodoc as xd

# `from_file` automatically resolves packages by looking in each parent
# directory of the given path to check for required ROS packages (as marked by
# a package.xml file)
doc = xd.XacroDoc.from_file("robot.urdf.xacro")

# if you want to disable this, pass `walk_up=False`:
doc = xd.XacroDoc.from_file("robot.urdf.xacro", walk_up=False)

# we can also tell xacrodoc to walk up a directory tree manually
xd.packages.walk_up_from("some/other/path")

# or we can give paths to directories to search for packages
# packages can be located multiple levels deep from the specified directories,
# just like in a ROS workspace - the same package search logic is used (since
# we actually use rospkg under the hood)
xd.packages.look_in(["somewhere/I/keep/packages", "another/directory/with/packages"])

# you can bypass ROS conventions entirely by directly providing a dict of
# package names and paths; no package.xml is required to identify packages in
# this case
xd.packages.update_package_cache({"my_package": "path/to/my_package"})

Multiple URDFs

We can also build a URDF programmatically from multiple xacro files:

import xacrodoc as xd

# setup where to look for packages, if needed; for example:
xd.packages.look_in(["somewhere/I/keep/packages"])

# specify files to compose (using xacro include directives)
includes = ["robot_base.urdf.xacro", "robot_arm.urdf.xacro", "tool.urdf.xacro"]
doc = xd.XacroDoc.from_includes(includes)

# includes can also use $(find ...) directives:
includes = [
    "$(find my_ros_package)/urdf/robot_base.urdf.xacro",
    "$(find another_ros_package)/urdf/robot_arm.urdf.xacro",
    "tool.urdf.xacro"
]
doc = xd.XacroDoc.from_includes(includes)

Substitution arguments

We can also pass in substitution arguments to xacro files. For example, suppose our file robot.urdf.xacro contains the directive <xacro:arg name="mass" default="1"/>. On the command line, we could write

xacro robot_base.urdf.xacro -o robot_base.urdf mass:=2

to set the mass value. Programmatically, we do

from xacrodoc import XacroDoc

doc = XacroDoc.from_file("robot.urdf.xacro", subargs={"mass": "2"})

Resolving file names with respect to packages

Finally, one feature of URDF (not just xacro files) is that file names (e.g., for meshes) can be specified relative to a package by using

package://<pkg>/relative/path/to/mesh

syntax, which depends on ROS and is not supported by other non-ROS tools. xacrodoc automatically expands these paths out to full absolute paths, e.g.,

file:///abs/path/to/mesh

but this can be disabled by passing resolve_packages=False to the XacroDoc constructor.

Conversion to MJCF format

Mujoco has basic support for URDFs, but natively uses its own MJCF XML format. If you want to use Mujoco, you probably want to convert any xacro file you have to MJCF. When handling URDFs, Mujoco automatically converts absolute file paths for assets like mesh files to relative paths. xacrodoc makes it easy to disable this behaviour or automatically copy the assets to a local relative directory. Note that mujoco must be installed and importable for this to work. For example:

from xacrodoc import XacroDoc

doc = XacroDoc.from_file("input.urdf.xacro")

# keep absolute paths to assets
doc.to_mjcf_file("output.xml", strippath="false")

# copy all the referenced assets to the directory `assets` before conversion
# name collisions between assets with the same filename are automatically
# resolved by appending numbers as needed
doc.localize_assets("assets")
doc.to_mjcf_file("output.xml", strippath="true", meshdir="assets")

# if desired, one can also just produce an MJCF string:
xml = doc.to_mjcf_string()

Both to_mjcf_file and to_mjcf_string accept keyword arguments corresponding to the Mujoco URDF compiler extension.

MJCF conversion is also available in the command line tool (see below).

Command Line Usage

In addition to the Python API described above, this packages also includes a xacrodoc command line tool. It is similar to xacro but provides additional options; notably, directories in which to search for packages can be provided manually. Examples:

# compile and print to stdout
xacrodoc input.urdf.xacro

# compile and output to provided output file
xacrodoc input.urdf.xacro -o output.urdf

# provide directories in which to look for packages referenced in
# input.urdf.xacro (-d flag is needed before each one to disambiguate from
# substitution arguments); a common usecase would be providing the path to
# catkin workspaces
xacrodoc input.urdf.xacro -d ~/my_pkg_dir -d ~/my_other_pkg_dir

# alternatively, package name/path mappings can be directly supplied using the
# -p flag; := is used to separate the name and path:
xacrodoc input.urdf.xacro -p my_pkg:=~/path/to/my_pkg -p other_pkg:=/home/foo/packages/other_pkg

# substitution arguments also use := notation, like xacro
xacrodoc input.urdf.xacro mass:=1

# you can also remove file:// protocols prefixed to asset paths with -s or
# convert paths from absolute to relative using -r
xacrodoc input.urdf.xacro -s -r output.urdf

# convert to MJCF (requires Mujoco)
# the -c (short for --copy-assets-to) option with a directory name is required
# when converting to MJCF; it copies all assets to that directory and updates
# their file paths (-c is optional when converting to URDF)
# relative file paths are always used when converting to MJCF
xacrodoc input.urdf.xacro --mjcf -c assets -o output.xml

Development

Testing

Tests use pytest. Some tests depend on additional submodules, which you can clone using:

git submodule update --init --recursive

Then do:

cd tests
pytest .

To test the Mujoco MJCF features, you'll also need mujoco installed. To test against different Python versions, use:

# for example
uv run --locked --isolated --extra mujoco --python=3.9 pytest

Docs

Docs are built with sphinx. In the docs directory, run

make html

and the open build/html/index.html.

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

xacrodoc-1.3.1.tar.gz (37.1 MB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

xacrodoc-1.3.1-py3-none-any.whl (163.7 kB view details)

Uploaded Python 3

File details

Details for the file xacrodoc-1.3.1.tar.gz.

File metadata

  • Download URL: xacrodoc-1.3.1.tar.gz
  • Upload date:
  • Size: 37.1 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.18 {"installer":{"name":"uv","version":"0.9.18","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for xacrodoc-1.3.1.tar.gz
Algorithm Hash digest
SHA256 336dc92a17adde0ffb08277131e00f69dce0404c0bb533e267e0cc44e406fa89
MD5 785835c88fd024edd6c32675b1261593
BLAKE2b-256 4846b77214f8a5991c79f19907fe8814aeb36f90129e50d028d91cb08c192d0c

See more details on using hashes here.

File details

Details for the file xacrodoc-1.3.1-py3-none-any.whl.

File metadata

  • Download URL: xacrodoc-1.3.1-py3-none-any.whl
  • Upload date:
  • Size: 163.7 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.18 {"installer":{"name":"uv","version":"0.9.18","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for xacrodoc-1.3.1-py3-none-any.whl
Algorithm Hash digest
SHA256 92410769392d4a11905b6a5c01ee7c61aa9fd33c42f849dd674012082a0605a4
MD5 2e632d0590a83d2e472e14749fabfb75
BLAKE2b-256 6bad38d8ac77c1c3e834ce7be752dd73f306c1567274e1e9e2154600242be210

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page