mirror of
https://github.com/huggingface/lerobot.git
synced 2026-06-04 04:41:24 +00:00
* docs(processor): update docstrings batch_processor * docs(processor): update docstrings device_processor * docs(processor): update docstrings tokenizer_processor * update docstrings processor_act * update docstrings for pipeline_features * update docstrings for utils * update docstring for processor_diffusion * update docstrings factory * add docstrings to pi0 processor * add docstring to pi0fast processor * add docstring classifier processor * add docstring to sac processor * add docstring smolvla processor * add docstring to tdmpc processor * add docstring to vqbet processor * add docstrings to converters * add docstrings for delta_action_processor * add docstring to gym action processor * update hil processor * add docstring to joint obs processor * add docstring to migrate_normalize_processor * update docstrings normalize processor * update docstring normalize processor * update docstrings observation processor * update docstrings rename_processor * add docstrings robot_kinematic_processor * cleanup rl comments * add docstring to train.py * add docstring to teleoperate.py * add docstrings to phone_processor.py * add docstrings to teleop_phone.py * add docstrings to control_utils.py * add docstrings to visualization_utils.py --------- Co-authored-by: Pepijn <pepijn@huggingface.co>
88 lines
3.3 KiB
Python
88 lines
3.3 KiB
Python
#!/usr/bin/env python
|
|
|
|
# Copyright 2025 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.
|
|
from copy import deepcopy
|
|
from dataclasses import dataclass, field
|
|
from typing import Any
|
|
|
|
from lerobot.configs.types import PolicyFeature
|
|
|
|
from .pipeline import ObservationProcessorStep, ProcessorStepRegistry
|
|
|
|
|
|
@dataclass
|
|
@ProcessorStepRegistry.register(name="rename_processor")
|
|
class RenameProcessorStep(ObservationProcessorStep):
|
|
"""
|
|
A processor step that renames keys in an observation dictionary.
|
|
|
|
This step is useful for creating a standardized data interface by mapping keys
|
|
from an environment's format to the format expected by a LeRobot policy or
|
|
other downstream components.
|
|
|
|
Attributes:
|
|
rename_map: A dictionary mapping from old key names to new key names.
|
|
Keys present in an observation that are not in this map will
|
|
be kept with their original names.
|
|
"""
|
|
|
|
rename_map: dict[str, str] = field(default_factory=dict)
|
|
|
|
def observation(self, observation):
|
|
processed_obs = {}
|
|
for key, value in observation.items():
|
|
if key in self.rename_map:
|
|
processed_obs[self.rename_map[key]] = value
|
|
else:
|
|
processed_obs[key] = value
|
|
|
|
return processed_obs
|
|
|
|
def get_config(self) -> dict[str, Any]:
|
|
return {"rename_map": self.rename_map}
|
|
|
|
def transform_features(self, features: dict[str, PolicyFeature]) -> dict[str, PolicyFeature]:
|
|
"""Transforms:
|
|
- Each key in the observation that appears in `rename_map` is renamed to its value.
|
|
- Keys not in `rename_map` remain unchanged.
|
|
"""
|
|
return {self.rename_map.get(k, k): v for k, v in features.items()}
|
|
|
|
|
|
def rename_stats(stats: dict[str, dict[str, Any]], rename_map: dict[str, str]) -> dict[str, dict[str, Any]]:
|
|
"""
|
|
Renames the top-level keys in a statistics dictionary using a provided mapping.
|
|
|
|
This is a helper function typically used to keep normalization statistics
|
|
consistent with renamed observation or action features. It performs a defensive
|
|
deep copy to avoid modifying the original `stats` dictionary.
|
|
|
|
Args:
|
|
stats: A nested dictionary of statistics, where top-level keys are
|
|
feature names (e.g., `{"observation.state": {"mean": 0.5}}`).
|
|
rename_map: A dictionary mapping old feature names to new feature names.
|
|
|
|
Returns:
|
|
A new statistics dictionary with its top-level keys renamed. Returns an
|
|
empty dictionary if the input `stats` is empty.
|
|
"""
|
|
if not stats:
|
|
return {}
|
|
renamed: dict[str, dict[str, Any]] = {}
|
|
for old_key, sub_stats in stats.items():
|
|
new_key = rename_map.get(old_key, old_key)
|
|
renamed[new_key] = deepcopy(sub_stats) if sub_stats is not None else {}
|
|
return renamed
|