Skip to main content

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, a built-in octahedron preset, 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 MjSpec models for triangle-based truss structures.
  • Generate a built-in octahedron robot preset.
  • 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:

  • Only the "octahedron" named preset is included.
  • Custom robot definitions are supported through dictionaries, but there is not yet a registry of named robot presets.
  • 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.viewer is available.

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
python -m pip install -e ".[dev]"

The package requires Python 3.10 or newer and installs these runtime dependencies:

  • gymnasium
  • mujoco
  • numpy
  • scipy

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:

  • 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:

  1. Update version in pyproject.toml.
  2. Commit and push the change.
  3. Create and publish a GitHub Release for that commit.
  4. Confirm the Publish to PyPI workflow 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.

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")

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()

Open the passive MuJoCo viewer:

python -m mujoco_truss_gen.generate_mujoco_model

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()

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.

Model Generation Contract

Public generation helpers:

  • build_world() creates a base mujoco.MjSpec containing 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.
  • get_mujoco_spec("octahedron", realistic=False) builds the built-in octahedron preset.
  • get_mujoco_spec(node_dict, triangle_dict, realistic=False) builds a custom dictionary-defined truss.
  • get_octahedron_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.
  • save_xml(spec, filename) writes spec.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.
  • 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=False creates one world-body per node with slide joints on x, y, and z. This is the simpler abstract model and is useful for fast algorithm development.
  • realistic=True creates 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.MjSpec
  • mujoco.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).
  • truncated becomes true when max_steps is reached.
  • terminated becomes true when the normalized rigidity metric falls below critical_eig_threshold.
  • info includes reward components and critical_eig.

Action behavior:

  • MujocoTrussEnv sends clipped actuator controls directly in the MuJoCo actuator control range.
  • MujocoRelativeObsEnv expects actions in [-1, 1]; each action component changes the previous control by action * config.speed.
  • MujocoVelocityCommandEnv expects 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(), or step() 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

Publishing Releases

PyPI releases are immutable for a given version. Every code change that should be published must use a new version number in pyproject.toml.

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).

Release checklist:

  1. Update version in pyproject.toml.
  2. Run python -m pytest.
  3. Run python -m ruff check ..
  4. Run python -m ruff format --check ..
  5. Build distributions with python -m build.
  6. Upload with python -m twine upload dist/*.
  7. 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

Repository Layout

mujoco-truss-gen/
├── LICENSE
├── README.md
├── 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


Download files

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

Source Distribution

mujoco_truss_gen-0.1.1a0.tar.gz (26.2 kB view details)

Uploaded Source

Built Distribution

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

mujoco_truss_gen-0.1.1a0-py3-none-any.whl (27.1 kB view details)

Uploaded Python 3

File details

Details for the file mujoco_truss_gen-0.1.1a0.tar.gz.

File metadata

  • Download URL: mujoco_truss_gen-0.1.1a0.tar.gz
  • Upload date:
  • Size: 26.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for mujoco_truss_gen-0.1.1a0.tar.gz
Algorithm Hash digest
SHA256 e4a9538ea10417d8947cf9c67182623eb09abd18aa07202ed4ea73269a98aba7
MD5 e9ef5db42a9c15f3760fccf2e77e8b05
BLAKE2b-256 05c979571adb30d0e0bbf31185c551fd138efa6496fa72860b2a6d2f5672c64e

See more details on using hashes here.

Provenance

The following attestation bundles were made for mujoco_truss_gen-0.1.1a0.tar.gz:

Publisher: publish.yml on isaa-sudweeks/mujoco-truss-gen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file mujoco_truss_gen-0.1.1a0-py3-none-any.whl.

File metadata

File hashes

Hashes for mujoco_truss_gen-0.1.1a0-py3-none-any.whl
Algorithm Hash digest
SHA256 38f7de3108a236c46ef339e75821de8a1494ad0bcf3faca262b346ba5fda77d2
MD5 3284775697c7fcf46d183a76883ab568
BLAKE2b-256 3b1228e06b9aa234cc55b0ba33006b9fa1b4a4433d039f1943cc7e78143088cd

See more details on using hashes here.

Provenance

The following attestation bundles were made for mujoco_truss_gen-0.1.1a0-py3-none-any.whl:

Publisher: publish.yml on isaa-sudweeks/mujoco-truss-gen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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