Ethan Clark 685854ecbd add case handling with urdf
2025-03-25 16:10:11 -07:00

114 lines
3.6 KiB
Python

import os
import re
import xml.dom.minidom as minidom
import xml.etree.ElementTree as ET
"""
Fixes a specific path that uses the ROS-style 'package://' prefix.
The 'package://' prefix is used in URDF files to refer to files in the same package.
However, when we're not running in a ROS environment, the paths are not valid.
This function tries to find the absolute path of the mesh file.
If the mesh file is not found, the original path is used.
"""
def fix_path(path: str, robot_base_dir: str) -> str:
if path.startswith("package://"):
parts = path[len("package://") :].split("/", 1)
if len(parts) == 2:
package_name, rel_path = parts
# Try potential locations for the mesh with relative paths
potential_paths = [
os.path.join("meshes", rel_path),
os.path.join(package_name, rel_path),
rel_path,
]
for possible_rel_path in potential_paths:
# Check if file exists with this relative path
full_path = os.path.join(robot_base_dir, possible_rel_path)
if os.path.exists(full_path):
# Return the relative path from urdf directory
return os.path.join("..", possible_rel_path)
print(f"Failed to find mesh for package path: {path}")
return path
"""
Iterates through the URDF file and fixes the paths of all mesh files.
The URDF file is parsed and the mesh paths are modified in-place.
"""
def fix_mesh_paths(urdf_path: str, urdf_dir: str) -> str:
root = ET.parse(urdf_path).getroot()
try:
for mesh in root.findall(".//mesh"):
if "filename" in mesh.attrib:
mesh.attrib["filename"] = fix_path(mesh.attrib["filename"], urdf_dir)
except Exception as e:
print(f"Error fixing mesh paths: {e}")
raise e
fixed_path = urdf_path.replace(".urdf", "_fixed.urdf")
return root, fixed_path
"""
Formats the XML tree to be human-readable and writes it to a file.
"""
def format_xml(root: ET.Element, fixed_path: str) -> None:
xml_str = ET.tostring(root, encoding="utf-8")
dom = minidom.parseString(xml_str)
with open(fixed_path, "w", encoding="utf-8") as f:
# Write with nice indentation but remove extra whitespace
pretty_xml = dom.toprettyxml(indent=" ")
# Remove extra blank lines that minidom sometimes adds
pretty_xml = "\n".join(
[line for line in pretty_xml.split("\n") if line.strip()]
)
f.write(pretty_xml)
def handle_urdf(urdf_path: str) -> str:
assert os.path.exists(urdf_path), f"Invalid URDF path: {urdf_path}"
try:
urdf_dir = os.path.dirname(urdf_path)
if urdf_path.endswith("_fixed.urdf"):
print("URDF is already fixed. Skipping processing.")
fixed_path = urdf_path
elif os.path.exists(urdf_path.replace(".urdf", "_fixed.urdf")):
print("Fixed URDF already exists. Skipping processing.")
fixed_path = urdf_path.replace(".urdf", "_fixed.urdf")
else:
print("Processing URDF...")
robot_base_dir = os.path.dirname(urdf_dir)
root, fixed_path = fix_mesh_paths(urdf_path, robot_base_dir)
format_xml(root, fixed_path)
print(f"Successfully processed URDF: {fixed_path}")
except Exception as e:
print(f"Failed to process URDF: {e}")
raise e
return fixed_path
if __name__ == "__main__":
# Example usage
cwd = os.getcwd()
urdf_path = os.path.join(
cwd,
"urdf/SO_5DOF_ARM100_05d.SLDASM/urdf/SO_5DOF_ARM100_05d.SLDASM.urdf",
)
handle_urdf(urdf_path)