mirror of
https://github.com/luckyrobots/open_phantom.git
synced 2025-04-04 02:52:18 +00:00
114 lines
3.6 KiB
Python
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)
|