Tools for generating MuJoCo models and Gym-style environments for isoperimetric truss robots.
Project description
mujoco-truss-gen
mujoco-truss-gen is a Python package for generating MuJoCo models and
Gymnasium-style environments for triangle-based isoperimetric truss robots.
The package is intended for members of the isoperimetric robot research workflow who need a shared, installable source of MuJoCo robot models instead of copying model-generation code between reinforcement learning, planning, simulation, and optimization projects.
Project Status
This repository is an internal lab prototype. It has a working installable package, a small public API, built-in polyhedron presets, and tests that verify basic model generation and environment stepping. The API may still change before the package is treated as stable research infrastructure.
Current scope:
- Generate MuJoCo
MjSpecmodels for triangle-based truss structures. - Generate built-in octahedron and icosahedron robot presets.
- Build either an abstract per-node slide-joint model or a more realistic triangle-body model with connector balls for shared nodes.
- Add tendon actuators and perimeter constraints.
- Save generated MuJoCo XML.
- Wrap generated models in Gymnasium-compatible environments.
- Provide base, relative-observation, and velocity-command environment variants.
Known limitations:
- The named preset registry is intentionally small. Current presets are
"octahedron","icosahedron","solar_array", and"tetrahedron". - Custom robot definitions are supported through dictionaries.
- The default rewards are research defaults, not task-independent objectives.
- The environment classes are starting points. Most RL, planning, or optimization tasks should subclass or wrap them for task-specific observations, rewards, resets, and termination logic.
- The human viewer requires a Python environment where
mujoco.vieweris available. - It is only possible to create custom trusses that are made up of independent triangles. Right now it is not possible to create a truss structure that includes triangles that share edges.
Future work:
- Generate a valid MuJoCo model for a continuous tube structure without manually defining every active edge.
- Add rigid elements between sets of nodes to support structures such as the Treg Rover.
- Explore importing a STEP file and generating a feasible truss approximation.
Installation
After a release has been published to PyPI:
python -m pip install mujoco-truss-gen
To upgrade to the newest published version:
python -m pip install --upgrade mujoco-truss-gen
For local development from a clone:
git clone https://github.com/isaa-sudweeks/mujoco-truss-gen.git
cd mujoco-truss-gen
python3 -m venv .venv
source .venv/bin/activate
python -m pip install -e ".[dev]"
On systems where python is not available, use python3 for the commands in
this README.
The package requires Python 3.10 or newer and installs these runtime dependencies:
gymnasiummujoconumpyscipy
Publishing Releases
PyPI publishing is automated through the GitHub Actions workflow in
.github/workflows/publish.yml. The workflow runs when a GitHub Release is
published, and it can also be started manually from the GitHub Actions tab.
The workflow:
For small test releases, you can use a pre-release tag (e.g., 0.1.0a1).
For bug fixes or backwards-compatible changes, you can use a patch release tag (e.g., 0.1.1).
For new features or breaking changes, you can use a minor or major release tag (e.g., 0.2.0 or 1.0.0).
- Installs the package test dependencies.
- Runs
python -m pytest. - Builds the source distribution and wheel with
python -m build. - Checks the built distributions with
python -m twine check dist/*. - Publishes the distributions to PyPI.
The publish job uses PyPI Trusted Publishing, so the repository does not need a
long-lived PyPI API token in GitHub Secrets. PyPI must be configured to trust
this GitHub Actions workflow before the first automated release. For the
existing mujoco-truss-gen PyPI project, add a GitHub Actions trusted publisher
with these settings:
- PyPI project name:
mujoco-truss-gen - GitHub repository owner:
isaa-sudweeks - GitHub repository name:
mujoco-truss-gen - Workflow filename:
publish.yml - GitHub environment name:
pypi
To publish a new version:
- Update
versioninpyproject.toml. - Commit and push the change.
- Create and publish a GitHub Release for that commit.
- Confirm the
Publish to PyPIworkflow passes.
PyPI versions are immutable. If a release workflow fails after uploading a
version, fix the issue, bump version again, and publish a new release.
Publishing Releases Manually
Release checklist:
- Update
versioninpyproject.toml. - Run
python -m pytest. - Run
python -m ruff check .. - Run
python -m ruff format --check .. - Build distributions with
python -m build. - Upload with
python -m twine upload dist/*. - Verify installation in a clean environment with
python -m pip install mujoco-truss-gen.
Users update to the newest published package with:
python -m pip install --upgrade mujoco-truss-gen
Quick Start
Generate the built-in octahedron model:
from mujoco_truss_gen import get_mujoco_spec
spec = get_mujoco_spec("octahedron", realistic=False)
model = spec.compile()
Save the generated XML:
from mujoco_truss_gen import get_mujoco_spec, save_xml
spec = get_mujoco_spec("octahedron", realistic=False)
xml_path = save_xml(spec, "octahedron.xml")
Relative output paths are resolved from the current working directory. The
example above writes octahedron.xml into the directory where the script is
run.
Run one Gymnasium step:
import numpy as np
from mujoco_truss_gen import MujocoRelativeObsEnv, TrussEnvConfig, get_mujoco_spec
spec = get_mujoco_spec("octahedron", realistic=False)
env = MujocoRelativeObsEnv(
TrussEnvConfig(
model_source=spec,
max_steps=1_000,
nsubsteps=4,
speed=0.01,
)
)
obs, info = env.reset(seed=0)
action = np.zeros(env.action_space.shape, dtype=np.float32)
obs, reward, terminated, truncated, info = env.step(action)
env.close()
View a generated model programmatically using the passive viewer:
from mujoco_truss_gen import get_mujoco_spec, view
spec = get_mujoco_spec("octahedron", realistic=False)
view(spec)
[!NOTE] On macOS,
mujoco.viewer.launch_passiverequires the script to be run withmjpythoninstead of the standardpythonexecutable:mjpython your_script.py
Alternatively, open the built-in octahedron model from the command line:
python -m mujoco_truss_gen.generate_mujoco_model
You can also start from the included custom-truss example:
python examples/custom_truss.py
Defining a Custom Truss
Custom trusses are represented with two dictionaries.
node_dict maps node names to 3D positions:
node_dict = {
"node_1": [0.0, 0.0, 0.2],
"node_2": [0.8, 0.0, 0.2],
"node_3": [0.4, 0.7, 0.2],
}
triangle_dict maps triangle names to four node names:
triangle_dict = {
"triangle_1": ["node_1", "node_2", "node_3", "node_1"],
}
The first three names are the triangle vertices. The fourth name is the passive node for that triangle's perimeter constraint and must be one of the first three vertex names.
Build a model from those dictionaries:
from mujoco_truss_gen import get_mujoco_spec
spec = get_mujoco_spec(node_dict, triangle_dict, realistic=False)
model = spec.compile()
Continuous tube shapes can also be represented with a shape dictionary. Each shape defines a routed path and the route edges that should be actuated:
node_dict = {
"node_1": [0.0, 0.0, 0.1],
"node_2": [1.0, 0.0, 0.1],
"node_3": [0.5, 0.8660, 0.1],
"node_4": [0.5, np.sqrt(3) / 6, 0.1 + np.sqrt(2 / 3)],
}
shape_dict = {
"path_1": {
"route": ["node_1", "node_2", "node_4", "node_3"],
"active_edges": [
["node_1", "node_2"],
["node_4", "node_3"],
],
},
"path_2": {
"route": ["node_2", "node_3", "node_1", "node_4"],
"active_edges": [
["node_2", "node_3"],
["node_1", "node_4"],
],
},
}
spec = get_mujoco_spec(node_dict, shape_dict, realistic=False)
model = spec.compile()
The route creates one edge tendon for each adjacent node pair in the path.
Active edges receive actuators. A route-length equality constraint keeps the
total routed length constant, so non-actuated passive edges absorb length
changes caused by the active edges. Routed shape dictionaries currently support
realistic=False.
get_mujoco_spec() and build_triangle() treat caller-provided dictionaries as
read-only inputs. The realistic builder clones shared nodes internally, but it
does not mutate the original node_dict or triangle_dict passed by the caller.
Graph Neural Network (GNN) Utilities
You can extract node features and edge indices directly from a spec or model, formatted specifically for use with PyTorch Geometric (torch_geometric.data.Data):
import torch
from mujoco_truss_gen import get_mujoco_spec, get_edge_index, get_node_features
spec = get_mujoco_spec("octahedron", realistic=False)
# Get edge indices in PyG COO format: shape (2, num_directed_edges)
edge_index = get_edge_index(spec)
# Get node features (position and velocity): shape (num_nodes, 6)
node_features = get_node_features(spec)
# Convert to PyTorch tensors
edge_index_tensor = torch.from_numpy(edge_index)
x_tensor = torch.from_numpy(node_features)
Model Generation Contract
Public generation helpers:
build_world()creates a basemujoco.MjSpeccontaining a ground plane and top light.build_triangle(spec, node_dict, triangle_dict, realistic=False)adds truss bodies, sites, tendons, actuators, and perimeter constraints to an existing spec.build_shapes(spec, node_dict, shape_dict, realistic=False)adds routed continuous-tube shapes with per-edge tendons and route-length constraints.get_mujoco_spec("octahedron", realistic=False)and the other names inPRESETSbuild built-in presets.get_mujoco_spec(node_dict, triangle_dict, realistic=False)builds a custom dictionary-defined truss.get_mujoco_spec(node_dict, shape_dict, realistic=False)builds a routed continuous-tube shape model.get_octahedron_definition()returns fresh node and triangle dictionaries for the built-in preset.get_icosahedron_definition()returns fresh node and triangle dictionaries for the built-in preset.get_perimeter(node_dict, triangle_dict)computes each triangle perimeter from the first three vertices.get_edge_index(source)extracts structural connectivity into a PyTorch Geometric compatible(2, E)numpy array.get_node_features(source)extracts node positions and velocities into a PyTorch Geometric compatible(N, 6)numpy array.save_xml(spec, filename)writesspec.to_xml()to disk and returns the resolved path.view(spec)compiles and opens the generated model in MuJoCo's passive viewer.
Input expectations:
- Node names should be unique strings. Names beginning with
node_are required for the built-in metadata and environment helpers. - Node positions must be 3D numeric sequences.
- Triangle entries must contain exactly the three vertex nodes plus one passive node.
- The passive node must appear in that triangle's first three vertices.
- Custom triangle definitions are validated before MuJoCo objects are created, and validation errors name the node, triangle, or shape entry that needs to be fixed.
- Shape entries must contain
routeandactive_edgeskeys. - Shape routes must contain at least two node names. Each active edge must be an adjacent pair in the route.
- The builder helpers should be used when environment rigidity and slip helpers are needed, because those helpers infer structure from generated body, site, tendon, and actuator names.
Model modes:
realistic=Falsecreates one world-body per node with slide joints onx,y, andz. This is the simpler abstract model and is useful for fast algorithm development.realistic=Truecreates triangle bodies, clones shared triangle nodes inside the generated model, and connects shared vertices through connector balls. This is intended to better represent the triangle-module structure.
Environment Contract
The environment constructors accept any of these model sources:
mujoco.MjSpecmujoco.MjModel- XML string
- path to an XML file
TrussEnvConfig
Available environments:
MujocoTrussEnv: base environment with tendon lengths, tendon velocities, center-of-mass position, and center-of-mass velocity in the observation.MujocoRelativeObsEnv: relative node-position observations and normalized actuator delta actions.MujocoVelocityCommandEnv: relative observations with direct velocity command actions.
Shared configuration is provided by TrussEnvConfig:
from mujoco_truss_gen import TrussEnvConfig
config = TrussEnvConfig(
model_source=spec,
max_steps=10_000,
nsubsteps=1,
speed=0.01,
forward_weight=5.0,
energy_weight=0.005,
alive_bonus=0.1,
rigidity_weight=0.5,
slip_weight=0.1,
critical_eig_threshold=0.03,
slip_height=0.2,
control_noise_std=0.0,
control_noise_relative=True,
runtime_apply_control_noise=False,
)
Step/reset behavior:
reset(seed=...)follows the Gymnasium API and returns(obs, info).step(action)returns(obs, reward, terminated, truncated, info).truncatedbecomes true whenmax_stepsis reached.terminatedbecomes true when the normalized rigidity metric falls belowcritical_eig_threshold.infoincludes reward components andcritical_eig.
Action behavior:
MujocoTrussEnvsends clipped actuator controls directly in the MuJoCo actuator control range.MujocoRelativeObsEnvexpects actions in[-1, 1]; each action component changes the previous control byaction * config.speed.MujocoVelocityCommandEnvexpects actions in[-config.speed, config.speed]and sends those values directly.
Reward behavior:
- The default reward combines forward velocity, alive bonus, energy penalty, rigidity reward, and slip penalty.
- These defaults are provided for experimentation, not as a canonical objective for every isoperimetric robot task.
- Custom tasks should subclass an environment and override
_get_obs(),_compute_reward(),reset(), orstep()as needed.
Rendering:
render_mode="rgb_array"returns a rendered NumPy RGB image.render_mode="human"opens a passive MuJoCo viewer when the local MuJoCo viewer module is available.
Development
Set up a development environment:
python -m pip install -e ".[dev]"
Run tests:
python -m pytest
Run linting and formatting checks:
python -m ruff check .
python -m ruff format --check .
Build a local distribution:
python -m build
Repository Layout
mujoco-truss-gen/
├── LICENSE
├── README.md
├── examples/
│ └── custom_truss.py
├── pyproject.toml
├── tests/
│ └── test_envs.py
└── src/
├── generate_mujoco_model.py
└── mujoco_truss_gen/
├── __init__.py
├── base_env.py
├── generate_mujoco_model.py
├── relative_observation_env.py
├── velocity_command_env.py
└── mujoco_model/
├── bodies.py
├── builders.py
├── constants.py
├── constraints.py
├── geometry.py
├── io_viewer.py
├── model.py
├── model_types.py
├── presets.py
└── tendons.py
Citation
There is no formal citation for this package yet.
License
This project is distributed under the BSD-3-Clause license. See LICENSE for
the full license text.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file mujoco_truss_gen-0.3.1.tar.gz.
File metadata
- Download URL: mujoco_truss_gen-0.3.1.tar.gz
- Upload date:
- Size: 33.4 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
c0c3aa93435fdffa6ff0a5efb93744055cf956760c44006af29b4f6c4a67e56f
|
|
| MD5 |
3e8cac14399bf10df52676dde72fe021
|
|
| BLAKE2b-256 |
12dc448e94b1b19e1e6708f45bc76e625c8e6b0f1de9f501689394aa3db21d88
|
Provenance
The following attestation bundles were made for mujoco_truss_gen-0.3.1.tar.gz:
Publisher:
publish.yml on isaa-sudweeks/mujoco-truss-gen
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mujoco_truss_gen-0.3.1.tar.gz -
Subject digest:
c0c3aa93435fdffa6ff0a5efb93744055cf956760c44006af29b4f6c4a67e56f - Sigstore transparency entry: 1454662089
- Sigstore integration time:
-
Permalink:
isaa-sudweeks/mujoco-truss-gen@642f12a64efde5a8a40aa5313827e4530cde5ad9 -
Branch / Tag:
refs/tags/v0.3.1 - Owner: https://github.com/isaa-sudweeks
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@642f12a64efde5a8a40aa5313827e4530cde5ad9 -
Trigger Event:
release
-
Statement type:
File details
Details for the file mujoco_truss_gen-0.3.1-py3-none-any.whl.
File metadata
- Download URL: mujoco_truss_gen-0.3.1-py3-none-any.whl
- Upload date:
- Size: 32.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
19144c5769de50195c9c3fdde9e644c33514e224865947c08c9b465a343b3983
|
|
| MD5 |
a4471b8c367355be130c18c604eb4e1c
|
|
| BLAKE2b-256 |
40f1f7d7f0551ee1f05214810ba06310067b2d1411f81b8420297cdfae5f777b
|
Provenance
The following attestation bundles were made for mujoco_truss_gen-0.3.1-py3-none-any.whl:
Publisher:
publish.yml on isaa-sudweeks/mujoco-truss-gen
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mujoco_truss_gen-0.3.1-py3-none-any.whl -
Subject digest:
19144c5769de50195c9c3fdde9e644c33514e224865947c08c9b465a343b3983 - Sigstore transparency entry: 1454662153
- Sigstore integration time:
-
Permalink:
isaa-sudweeks/mujoco-truss-gen@642f12a64efde5a8a40aa5313827e4530cde5ad9 -
Branch / Tag:
refs/tags/v0.3.1 - Owner: https://github.com/isaa-sudweeks
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@642f12a64efde5a8a40aa5313827e4530cde5ad9 -
Trigger Event:
release
-
Statement type: