mirror of
https://github.com/huggingface/lerobot.git
synced 2026-05-30 18:31:25 +00:00
* adding Isaaclab Arena from collab
* adding into lerobot-eval
* minor modification
* added bash script for env setup
* setups
* fix applauncher not getting the arguments
* data conversion, train and eval smolvla
* fixed imports
* clean-up
* added test suits & clean up - wip
* fixed video recording
* clean-up
* hub integration working
* clean-up
* added kwargs
* Revert "added kwargs"
This reverts commit 9b445356385d0707655cf04d02be058b25138119.
* added kwargs
* clean-up
* cleaned unused function
* added logging
* docs
* cleaned up IsaaclabArenaEnv
* clean-up
* clean-up
* clean up
* added tests
* minor clean-up
* fix: support for state based envs
* feat(envs): Add NVIDIA IsaacLab Arena integration with LeRobot for policy evaluation at scale
* feat(envs): Add IsaacLab Arena integration for policy evaluation
Integrate NVIDIA IsaacLab Arena with LeRobot to enable GPU-accelerated
simulation through the EnvHub infrastructure.
This enables:
- Training imitation learning policies (PI0, SmolVLA, etc.)
- Evaluating trained policies in with IsaacLab Arena
The implementation adds:
- IsaaclabArenaEnv config with Arena-specific parameters
- IsaaclabArenaProcessorStep for observation processing
- Hub loading from nvkartik/isaaclab-arena-envs repository
- Video recording support
Available environments include GR1 microwave manipulation, Galileo
pick-and-place, G1 loco-manipulation, and button pressing tasks.
Datasets: nvkartik/Arena-GR1-Manipulation-Task
Policies: nvkartik/pi05-arena-gr1-microwave,
nvkartik/smolvla-arena-gr1-microwave
* added isaaclab arena wrapper and corresponding tests
* added error handling
* renamed wrapper file: isaaclab_arena to isaaclab
* added extra kwarg changes
* adjustments for hub envs
* correct class name in test file
* fixed parsing of env_kwargs
* tested end to end
* removed unused code
* refactor design
* shifted IsaacLab to hub
* removed IsaacLab tests
* docs: Add LW-BenchHub evaluation instructions
* docs: Add LW-BenchHub evaluation instructions
* docs diet
* minor edits to texts
* IL Arena commit hash
* update links
* minor edits
* fix numpy version after install of lerobot
* links update
* valideated on vanilla brev
* docs: Add LW-BenchHub evaluation instructions
* remove kwargs from all make_env calls
* remove kwargs from all make_env calls
* fix LW table and indentations
* remove environment list from docs
* docs: Update lw-benchhub eval config in envhub docs
* removing kwargs
* removed extra line
* ensure pinocchio install for lightwheel + add lightwheel website link
* remove env_kwargs
* no default empty value for hub_path
* not using assert method
* remove env_processor defaults
* revert and adding default "" value for hub_path
* pinning down packages versions
* explicit None value for hub_path
* Update src/lerobot/configs/eval.py
Co-authored-by: Jade Choghari <chogharijade@gmail.com>
Signed-off-by: Lior Ben Horin <liorbenhorin@gmail.com>
* corrected formatting
* corrected job_name var in config
* updated docs and namespace
* updated namespace
* updated docs
* updated docs
* added hardware requirements
* updated docs
---------
Signed-off-by: Lior Ben Horin <liorbenhorin@gmail.com>
Co-authored-by: lbenhorin <lbenhorin@nvidia.com>
Co-authored-by: Lior Ben Horin <liorbenhorin@gmail.com>
Co-authored-by: Jade Choghari <chogharijade@gmail.com>
Co-authored-by: Steven Palma <imstevenpmwork@ieee.org>
Co-authored-by: tianheng.wu <tianheng.wu@lightwheel.ai>
220 lines
8.9 KiB
Python
220 lines
8.9 KiB
Python
#!/usr/bin/env python
|
|
|
|
# Copyright 2024 The HuggingFace Inc. team. All rights reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
import importlib
|
|
from typing import Any
|
|
|
|
import gymnasium as gym
|
|
from gymnasium.envs.registration import registry as gym_registry
|
|
|
|
from lerobot.configs.policies import PreTrainedConfig
|
|
from lerobot.envs.configs import AlohaEnv, EnvConfig, HubEnvConfig, IsaaclabArenaEnv, LiberoEnv, PushtEnv
|
|
from lerobot.envs.utils import _call_make_env, _download_hub_file, _import_hub_module, _normalize_hub_result
|
|
from lerobot.policies.xvla.configuration_xvla import XVLAConfig
|
|
from lerobot.processor import ProcessorStep
|
|
from lerobot.processor.env_processor import IsaaclabArenaProcessorStep, LiberoProcessorStep
|
|
from lerobot.processor.pipeline import PolicyProcessorPipeline
|
|
|
|
|
|
def make_env_config(env_type: str, **kwargs) -> EnvConfig:
|
|
if env_type == "aloha":
|
|
return AlohaEnv(**kwargs)
|
|
elif env_type == "pusht":
|
|
return PushtEnv(**kwargs)
|
|
elif env_type == "libero":
|
|
return LiberoEnv(**kwargs)
|
|
else:
|
|
raise ValueError(f"Policy type '{env_type}' is not available.")
|
|
|
|
|
|
def make_env_pre_post_processors(
|
|
env_cfg: EnvConfig,
|
|
policy_cfg: PreTrainedConfig,
|
|
) -> tuple[
|
|
PolicyProcessorPipeline[dict[str, Any], dict[str, Any]],
|
|
PolicyProcessorPipeline[dict[str, Any], dict[str, Any]],
|
|
]:
|
|
"""
|
|
Create preprocessor and postprocessor pipelines for environment observations.
|
|
|
|
This function creates processor pipelines that transform raw environment
|
|
observations and actions. By default, it returns identity processors that do nothing.
|
|
For specific environments like LIBERO, it adds environment-specific processing steps.
|
|
|
|
Args:
|
|
env_cfg: The configuration of the environment.
|
|
|
|
Returns:
|
|
A tuple containing:
|
|
- preprocessor: Pipeline that processes environment observations
|
|
- postprocessor: Pipeline that processes environment outputs (currently identity)
|
|
"""
|
|
# Preprocessor and Postprocessor steps are Identity for most environments
|
|
preprocessor_steps: list[ProcessorStep] = []
|
|
postprocessor_steps: list[ProcessorStep] = []
|
|
if isinstance(policy_cfg, XVLAConfig):
|
|
from lerobot.policies.xvla.processor_xvla import make_xvla_libero_pre_post_processors
|
|
|
|
return make_xvla_libero_pre_post_processors()
|
|
|
|
# For LIBERO environments, add the LiberoProcessorStep to preprocessor
|
|
if isinstance(env_cfg, LiberoEnv) or "libero" in env_cfg.type:
|
|
preprocessor_steps.append(LiberoProcessorStep())
|
|
|
|
# For Isaaclab Arena environments, add the IsaaclabArenaProcessorStep
|
|
if isinstance(env_cfg, IsaaclabArenaEnv) or "isaaclab_arena" in env_cfg.type:
|
|
# Parse comma-separated keys (handle None for state-based policies)
|
|
if env_cfg.state_keys:
|
|
state_keys = tuple(k.strip() for k in env_cfg.state_keys.split(",") if k.strip())
|
|
else:
|
|
state_keys = ()
|
|
if env_cfg.camera_keys:
|
|
camera_keys = tuple(k.strip() for k in env_cfg.camera_keys.split(",") if k.strip())
|
|
else:
|
|
camera_keys = ()
|
|
if not state_keys and not camera_keys:
|
|
raise ValueError("At least one of state_keys or camera_keys must be specified.")
|
|
preprocessor_steps.append(
|
|
IsaaclabArenaProcessorStep(
|
|
state_keys=state_keys,
|
|
camera_keys=camera_keys,
|
|
)
|
|
)
|
|
|
|
preprocessor = PolicyProcessorPipeline(steps=preprocessor_steps)
|
|
postprocessor = PolicyProcessorPipeline(steps=postprocessor_steps)
|
|
|
|
return preprocessor, postprocessor
|
|
|
|
|
|
def make_env(
|
|
cfg: EnvConfig | str,
|
|
n_envs: int = 1,
|
|
use_async_envs: bool = False,
|
|
hub_cache_dir: str | None = None,
|
|
trust_remote_code: bool = False,
|
|
) -> dict[str, dict[int, gym.vector.VectorEnv]]:
|
|
"""Makes a gym vector environment according to the config or Hub reference.
|
|
|
|
Args:
|
|
cfg (EnvConfig | str): Either an `EnvConfig` object describing the environment to build locally,
|
|
or a Hugging Face Hub repository identifier (e.g. `"username/repo"`). In the latter case,
|
|
the repo must include a Python file (usually `env.py`).
|
|
n_envs (int, optional): The number of parallelized env to return. Defaults to 1.
|
|
use_async_envs (bool, optional): Whether to return an AsyncVectorEnv or a SyncVectorEnv. Defaults to
|
|
False.
|
|
hub_cache_dir (str | None): Optional cache path for downloaded hub files.
|
|
trust_remote_code (bool): **Explicit consent** to execute remote code from the Hub.
|
|
Default False — must be set to True to import/exec hub `env.py`.
|
|
Raises:
|
|
ValueError: if n_envs < 1
|
|
ModuleNotFoundError: If the requested env package is not installed
|
|
|
|
Returns:
|
|
dict[str, dict[int, gym.vector.VectorEnv]]:
|
|
A mapping from suite name to indexed vectorized environments.
|
|
- For multi-task benchmarks (e.g., LIBERO): one entry per suite, and one vec env per task_id.
|
|
- For single-task environments: a single suite entry (cfg.type) with task_id=0.
|
|
|
|
"""
|
|
# if user passed a hub id string (e.g., "username/repo", "username/repo@main:env.py")
|
|
# simplified: only support hub-provided `make_env`
|
|
# TODO: (jadechoghari): deprecate string API and remove this check
|
|
if isinstance(cfg, str):
|
|
hub_path: str | None = cfg
|
|
elif isinstance(cfg, HubEnvConfig):
|
|
hub_path = cfg.hub_path
|
|
else:
|
|
hub_path = None
|
|
|
|
# If hub_path is set, download and call hub-provided `make_env`
|
|
if hub_path:
|
|
# _download_hub_file will raise the same RuntimeError if trust_remote_code is False
|
|
repo_id, file_path, local_file, revision = _download_hub_file(
|
|
hub_path, trust_remote_code, hub_cache_dir
|
|
)
|
|
|
|
# import and surface clear import errors
|
|
module = _import_hub_module(local_file, repo_id)
|
|
|
|
# call the hub-provided make_env
|
|
env_cfg = None if isinstance(cfg, str) else cfg
|
|
raw_result = _call_make_env(module, n_envs=n_envs, use_async_envs=use_async_envs, cfg=env_cfg)
|
|
|
|
# normalize the return into {suite: {task_id: vec_env}}
|
|
return _normalize_hub_result(raw_result)
|
|
|
|
# At this point, cfg must be an EnvConfig (not a string) since hub_path would have been set otherwise
|
|
if isinstance(cfg, str):
|
|
raise TypeError("cfg should be an EnvConfig at this point")
|
|
|
|
if n_envs < 1:
|
|
raise ValueError("`n_envs` must be at least 1")
|
|
|
|
env_cls = gym.vector.AsyncVectorEnv if use_async_envs else gym.vector.SyncVectorEnv
|
|
|
|
if "libero" in cfg.type:
|
|
from lerobot.envs.libero import create_libero_envs
|
|
|
|
if cfg.task is None:
|
|
raise ValueError("LiberoEnv requires a task to be specified")
|
|
|
|
return create_libero_envs(
|
|
task=cfg.task,
|
|
n_envs=n_envs,
|
|
camera_name=cfg.camera_name,
|
|
init_states=cfg.init_states,
|
|
gym_kwargs=cfg.gym_kwargs,
|
|
env_cls=env_cls,
|
|
control_mode=cfg.control_mode,
|
|
episode_length=cfg.episode_length,
|
|
)
|
|
elif "metaworld" in cfg.type:
|
|
from lerobot.envs.metaworld import create_metaworld_envs
|
|
|
|
if cfg.task is None:
|
|
raise ValueError("MetaWorld requires a task to be specified")
|
|
|
|
return create_metaworld_envs(
|
|
task=cfg.task,
|
|
n_envs=n_envs,
|
|
gym_kwargs=cfg.gym_kwargs,
|
|
env_cls=env_cls,
|
|
)
|
|
|
|
if cfg.gym_id not in gym_registry:
|
|
print(f"gym id '{cfg.gym_id}' not found, attempting to import '{cfg.package_name}'...")
|
|
try:
|
|
importlib.import_module(cfg.package_name)
|
|
except ModuleNotFoundError as e:
|
|
raise ModuleNotFoundError(
|
|
f"Package '{cfg.package_name}' required for env '{cfg.type}' not found. "
|
|
f"Please install it or check PYTHONPATH."
|
|
) from e
|
|
|
|
if cfg.gym_id not in gym_registry:
|
|
raise gym.error.NameNotFound(
|
|
f"Environment '{cfg.gym_id}' not registered even after importing '{cfg.package_name}'."
|
|
)
|
|
|
|
def _make_one():
|
|
return gym.make(cfg.gym_id, disable_env_checker=cfg.disable_env_checker, **(cfg.gym_kwargs or {}))
|
|
|
|
vec = env_cls([_make_one for _ in range(n_envs)], autoreset_mode=gym.vector.AutoresetMode.SAME_STEP)
|
|
|
|
# normalize to {suite: {task_id: vec_env}} for consistency
|
|
suite_name = cfg.type # e.g., "pusht", "aloha"
|
|
return {suite_name: {0: vec}}
|