diff --git a/aip_urdf_compiler/CMakeLists.txt b/aip_urdf_compiler/CMakeLists.txt
new file mode 100644
index 00000000..459dd61d
--- /dev/null
+++ b/aip_urdf_compiler/CMakeLists.txt
@@ -0,0 +1,19 @@
+cmake_minimum_required(VERSION 3.5)
+project(aip_urdf_compiler)
+
+find_package(ament_cmake_auto REQUIRED)
+
+ament_auto_find_build_dependencies()
+
+# Install cmake directory
+install(
+ DIRECTORY cmake templates scripts
+ DESTINATION share/${PROJECT_NAME}
+)
+
+# Export the package's share directory path
+
+# Add the config extras
+ament_package(
+ CONFIG_EXTRAS "cmake/aip_cmake_urdf_compile.cmake"
+)
diff --git a/aip_urdf_compiler/cmake/aip_cmake_urdf_compile.cmake b/aip_urdf_compiler/cmake/aip_cmake_urdf_compile.cmake
new file mode 100644
index 00000000..70c13c47
--- /dev/null
+++ b/aip_urdf_compiler/cmake/aip_cmake_urdf_compile.cmake
@@ -0,0 +1,33 @@
+
+
+
+macro(aip_cmake_urdf_compile)
+ # Set the correct paths
+ find_package(PythonInterp REQUIRED) # cspell: ignore Interp
+ set(aip_urdf_compiler_BASE_DIR "${aip_urdf_compiler_DIR}/../")
+ set(PYTHON_SCRIPT "${aip_urdf_compiler_BASE_DIR}/scripts/compile_xacro.py")
+ set(PYTHON_TEMPLATE_DIRECTORY "${aip_urdf_compiler_BASE_DIR}/templates")
+ set(PYTHON_CALIBRATION_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/config")
+ set(PYTHON_XACRO_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/urdf")
+
+ message(STATUS "PYTHON_SCRIPT path: ${PYTHON_SCRIPT}")
+ message(STATUS "PYTHON_TEMPLATE_DIRECTORY path: ${PYTHON_TEMPLATE_DIRECTORY}")
+
+ # Verify that the required files exist
+ if(NOT EXISTS "${PYTHON_SCRIPT}")
+ message(FATAL_ERROR "Could not find compile_xacro.py at ${PYTHON_SCRIPT}")
+ endif()
+
+ # Create a custom command to run the Python script
+ add_custom_target(xacro_compilation ALL
+ DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/python_script_run_flag
+ )
+
+ add_custom_command(
+ OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/python_script_run_flag
+ COMMAND ${PYTHON_EXECUTABLE} ${PYTHON_SCRIPT} ${PYTHON_TEMPLATE_DIRECTORY} ${PYTHON_CALIBRATION_DIRECTORY} ${PYTHON_XACRO_DIRECTORY} ${PROJECT_NAME}
+ WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
+ DEPENDS ${PYTHON_SCRIPT}
+ COMMENT "Running Python script for URDF creation"
+ )
+endmacro()
diff --git a/aip_urdf_compiler/package.xml b/aip_urdf_compiler/package.xml
new file mode 100644
index 00000000..c6355e42
--- /dev/null
+++ b/aip_urdf_compiler/package.xml
@@ -0,0 +1,17 @@
+
+
+ aip_urdf_compiler
+ 0.1.0
+ The aip_urdf_compiler package
+
+ Yukihiro Saito
+ Yuxuan Liu
+ Apache 2
+
+ ament_cmake_auto
+ python3-jinja2
+
+
+ ament_cmake
+
+
diff --git a/aip_urdf_compiler/scripts/compile_xacro.py b/aip_urdf_compiler/scripts/compile_xacro.py
new file mode 100644
index 00000000..a82a2f6e
--- /dev/null
+++ b/aip_urdf_compiler/scripts/compile_xacro.py
@@ -0,0 +1,484 @@
+#!/usr/bin/python3
+"""
+XACRO Compiler Script.
+
+This script compiles XACRO files for robot sensor configurations. It processes calibration data
+from YAML files and generates XACRO files that define the sensor transforms and configurations
+for a robot's URDF description.
+
+The script handles various types of sensors including cameras, IMUs, LiDARs (Velodyne, Pandar, Livox),
+and radar units, generating appropriate XACRO macros for each sensor type.
+"""
+
+import enum
+import functools
+import os
+from typing import Callable
+from typing import Dict
+from typing import Union
+
+from jinja2 import Template
+import yaml
+
+
+def load_yaml(file_path: str) -> Dict:
+ """
+ Load and parse a YAML file.
+
+ Args:
+ file_path (str): Path to the YAML file
+
+ Returns:
+ Dict: Parsed YAML content or None if parsing fails
+ """
+ with open(file_path, "r") as stream:
+ try:
+ return yaml.safe_load(stream)
+ except yaml.YAMLError as exc:
+ print(exc)
+ return None
+
+
+class Transformation:
+ """
+ Represents a coordinate transformation between two frames.
+
+ Stores translation (x,y,z) and rotation (roll,pitch,yaw) parameters
+ along with frame information and sensor type.
+ """
+
+ def __init__(self, transformation: Dict, base_frame: str, child_frame: str):
+ """
+ Initialize a transformation from a dictionary of parameters.
+
+ Args:
+ transformation (Dict): Dictionary containing transformation parameters
+ base_frame (str): Name of the parent/base frame
+ child_frame (str): Name of the child frame
+
+ Raises:
+ KeyError: If required transformation parameters are missing
+ """
+ try:
+ self.x = transformation["x"]
+ self.y = transformation["y"]
+ self.z = transformation["z"]
+ self.roll = transformation["roll"]
+ self.pitch = transformation["pitch"]
+ self.yaw = transformation["yaw"]
+ self.base_frame = base_frame
+ self.child_frame = child_frame
+ self.type: str = transformation.get("type", "")
+
+ self.name = self.child_frame.replace("_base_link", "").replace("_link", "")
+
+ if len(self.type) == 0:
+ self.type = determine_link_type(self.name)
+
+ self.frame_id: str = transformation.get("frame_id", "")
+ if len(self.frame_id) == 0:
+ if (
+ "pandar" in self.type
+ or "livox" in self.type
+ or "camera" in self.type
+ or "vls" in self.type.lower()
+ or "vlp" in self.type.lower()
+ ):
+ # For common sensor descriptions, LiDAR and camera macros will automatically
+ # be attached with a "base_link" name
+ self.frame_id = self.name
+ else:
+ self.frame_id = self.child_frame
+
+ except KeyError as e:
+ print(f"Error: Key {e} not in transformation dictionary")
+ raise e
+
+ def serialize_single(self, key: str) -> str:
+ """
+ Generate a serialized string for a single transformation parameter.
+
+ Args:
+ key (str): Parameter key (x, y, z, roll, pitch, or yaw)
+
+ Returns:
+ str: Serialized parameter string for use in XACRO template
+ """
+ return f"${{calibration['{self.base_frame}']['{self.child_frame}']['{key}']}}"
+
+ def serialize(self) -> str:
+ """
+ Generate a complete serialized string for all transformation parameters.
+
+ Returns:
+ str: Complete serialized transformation string for XACRO template
+ """
+ return f"""
+ name=\"{self.name}\"
+ parent=\"{self.base_frame}\"
+ x=\"{self.serialize_single('x')}\"
+ y=\"{self.serialize_single('y')}\"
+ z=\"{self.serialize_single('z')}\"
+ roll=\"{self.serialize_single('roll')}\"
+ pitch=\"{self.serialize_single('pitch')}\"
+ yaw=\"{self.serialize_single('yaw')}\"
+ """
+
+
+class Calibration:
+ """
+ Represents a complete set of calibration data for all sensors.
+
+ Contains transformations for all sensors relative to a single base frame.
+ """
+
+ def __init__(self, calibration: Dict):
+ """
+ Initialize calibration data from a dictionary.
+
+ Args:
+ calibration (Dict): Dictionary containing calibration data
+
+ Raises:
+ AssertionError: If calibration format is invalid
+ """
+ self.base_dict: Dict = calibration
+ assert len(calibration.keys()) == 1, "Calibration file should have only one base frame"
+ assert isinstance(
+ list(calibration.keys())[0], str
+ ), "Calibration file should have only one base frame with key as a string"
+ self.base_frame: str = list(calibration.keys())[0]
+
+ assert isinstance(
+ calibration[self.base_frame], dict
+ ), "Calibration file should have only one base frame with value as a dictionary"
+
+ self.transforms: Dict[str, Transformation] = {}
+
+ for key in calibration[self.base_frame]:
+ assert isinstance(key, str), "child frames should be strings"
+ try:
+ self.transforms[key] = Transformation(
+ calibration[self.base_frame][key], self.base_frame, key
+ )
+ except KeyError as e:
+ print(f"Error: Key {e} not in calibration dictionary of {key}")
+ raise e
+
+
+class LinkType(enum.Enum):
+ """Enum class for the type of the link."""
+
+ CAMERA = "monocular_camera"
+ IMU = "imu"
+ LIVOX = "livox_horizon"
+ PANDAR_40P = "pandar_40p"
+ PANDAR_OT128 = "pandar_ot128"
+ PANDAR_XT32 = "pandar_xt32"
+ PANDAR_QT = "pandar_qt"
+ PANDAR_QT128 = "pandar_qt128"
+ VELODYNE16 = "VLP-16.urdf"
+ VLS128 = "VLS-128.urdf"
+ RADAR = "radar"
+ JOINT_UNITS = "units"
+
+
+def obtain_link_type(link: Transformation) -> LinkType:
+ """Output the LinkType of the target link."""
+ if len(link.type) > 0:
+ # use explicit type string to obtain link
+ link_type_lower = link.type.lower()
+
+ # Check each enum value for a match
+ for type_enum in LinkType:
+ if link_type_lower == type_enum.value.lower():
+ return type_enum
+ # if there is no match, or the type is not defined:
+ return determine_link_type(link.child_frame)
+
+
+def determine_link_type(link_name: str) -> LinkType:
+ """Produce a guess of the type of the link based on its name."""
+ if "cam" in link_name:
+ return LinkType.CAMERA
+
+ if "imu" in link_name or "gnss" in link_name:
+ return LinkType.IMU
+
+ if "livox" in link_name:
+ return LinkType.LIVOX
+
+ if "velodyne" in link_name:
+ if "top" in link_name:
+ return LinkType.VLS128
+ else:
+ return LinkType.VELODYNE16
+
+ if "radar" in link_name or "ars" in link_name:
+ return LinkType.RADAR
+
+ if "pandar_40p" in link_name:
+ return LinkType.PANDAR_40P
+
+ if "pandar_qt" in link_name:
+ return LinkType.PANDAR_QT
+
+ if "hesai_top" in link_name:
+ return LinkType.PANDAR_OT128
+
+ if "hesai_front" in link_name:
+ return LinkType.PANDAR_XT32
+
+ if "hesai" in link_name:
+ return LinkType.PANDAR_XT32
+
+ else:
+ print(f"Link type not found for {link_name}, suspected to be a joint unit")
+ return LinkType.JOINT_UNITS
+
+
+BASE_STRING = """"""
+
+VLD_STRING = """
+
+ """
+
+
+def base_string_func(macro_type: str, transform: Transformation) -> str:
+ if macro_type == "monocular_camera_macro":
+ extra = """fps=\"30\"
+ width=\"800\"
+ height=\"400\"
+ namespace=\"\"
+ fov=\"1.3\""""
+ elif macro_type == "imu_macro":
+ extra = """fps=\"100\"
+ namespace=\"\""""
+ else:
+ extra = ""
+ return BASE_STRING.format(
+ type=macro_type,
+ base_frame=transform.base_frame,
+ child_frame=transform.frame_id, # pandar
+ x=transform.serialize_single("x"),
+ y=transform.serialize_single("y"),
+ z=transform.serialize_single("z"),
+ roll=transform.serialize_single("roll"),
+ pitch=transform.serialize_single("pitch"),
+ yaw=transform.serialize_single("yaw"),
+ extra=extra,
+ )
+
+
+def VLP16_func(transform: Transformation) -> str:
+ return VLD_STRING.format(
+ type="VLP-16",
+ base_frame=transform.base_frame,
+ child_frame=transform.frame_id,
+ x=transform.serialize_single("x"),
+ y=transform.serialize_single("y"),
+ z=transform.serialize_single("z"),
+ roll=transform.serialize_single("roll"),
+ pitch=transform.serialize_single("pitch"),
+ yaw=transform.serialize_single("yaw"),
+ )
+
+
+def VLS128_func(transform: Transformation) -> str:
+ return VLD_STRING.format(
+ type="VLS-128",
+ base_frame=transform.base_frame,
+ child_frame=transform.frame_id,
+ x=transform.serialize_single("x"),
+ y=transform.serialize_single("y"),
+ z=transform.serialize_single("z"),
+ roll=transform.serialize_single("roll"),
+ pitch=transform.serialize_single("pitch"),
+ yaw=transform.serialize_single("yaw"),
+ )
+
+
+"""
+link_dicts maps the LinkType to its required include files and the template strings.
+including_file is the path to the required sub module xacro
+string_api is a function that outputs a template string from a transform
+"""
+
+link_dicts: Dict[LinkType, Dict[str, Union[str, Callable[[Transformation], str]]]] = {
+ LinkType.CAMERA: {
+ "including_file": "$(find camera_description)/urdf/monocular_camera.xacro",
+ "string_api": functools.partial(base_string_func, "monocular_camera_macro"),
+ },
+ LinkType.IMU: {
+ "including_file": "$(find imu_description)/urdf/imu.xacro",
+ "string_api": functools.partial(base_string_func, "imu_macro"),
+ },
+ LinkType.VELODYNE16: {
+ "including_file": "$(find velodyne_description)/urdf/VLP-16.urdf.xacro",
+ "string_api": VLP16_func,
+ },
+ LinkType.VLS128: {
+ "including_file": "$(find vls_description)/urdf/VLS-128.urdf.xacro",
+ "string_api": VLS128_func,
+ },
+ LinkType.PANDAR_40P: {
+ "including_file": "$(find pandar_description)/urdf/pandar_40p.xacro",
+ "string_api": functools.partial(base_string_func, "Pandar40P"),
+ },
+ LinkType.PANDAR_OT128: {
+ "including_file": "$(find pandar_description)/urdf/pandar_ot128.xacro",
+ "string_api": functools.partial(base_string_func, "PandarOT-128"),
+ },
+ LinkType.PANDAR_XT32: {
+ "including_file": "$(find pandar_description)/urdf/pandar_xt32.xacro",
+ "string_api": functools.partial(base_string_func, "PandarXT-32"),
+ },
+ LinkType.PANDAR_QT: {
+ "including_file": "$(find pandar_description)/urdf/pandar_qt.xacro",
+ "string_api": functools.partial(base_string_func, "PandarQT"),
+ },
+ LinkType.PANDAR_QT128: {
+ "including_file": "$(find pandar_description)/urdf/pandar_qt128.xacro",
+ "string_api": functools.partial(base_string_func, "PandarQT-128"),
+ },
+ LinkType.LIVOX: {
+ "including_file": "$(find livox_description)/urdf/livox_horizon.xacro",
+ "string_api": functools.partial(base_string_func, "livox_horizon_macro"),
+ },
+ LinkType.RADAR: {
+ "including_file": "$(find radar_description)/urdf/radar.xacro",
+ "string_api": functools.partial(base_string_func, "radar_macro"),
+ },
+ LinkType.JOINT_UNITS: {
+ "including_file": "{filename}.xacro",
+ },
+}
+
+
+def main(
+ template_directory: str,
+ calibration_directory: str,
+ output_directory: str,
+ project_name: str,
+):
+ os.makedirs(output_directory, exist_ok=True)
+ # Load the template
+ with open(os.path.join(template_directory, "sensors.xacro.template"), "r") as file:
+ base_template = Template(file.read())
+
+ # Render the template
+ calibration_path = os.path.join(calibration_directory, "sensors_calibration.yaml")
+ calib_yaml = load_yaml(calibration_path)
+ calib = Calibration(calib_yaml)
+
+ render_meta_data = {}
+ render_meta_data["default_config_path"] = f"$(find {project_name})/config"
+ render_meta_data["sensor_calibration_yaml_path"] = "$(arg config_dir)/sensors_calibration.yaml"
+ render_meta_data["sensor_units_includes"] = []
+ render_meta_data["sensor_units"] = []
+ render_meta_data["isolated_sensors_includes"] = []
+ render_meta_data["isolated_sensors"] = []
+
+ include_text = set()
+ sensor_items = []
+ for _, transform in calib.transforms.items():
+ link_type: LinkType = obtain_link_type(transform)
+ if link_type == LinkType.JOINT_UNITS:
+ render_meta_data["sensor_units_includes"].append(
+ link_dicts[link_type]["including_file"].format(filename=transform.name)
+ )
+ render_meta_data["sensor_units"].append(
+ {
+ "base_frame": transform.base_frame,
+ "child_frame": transform.child_frame,
+ "macro_name": f"{transform.name}_macro",
+ "name": transform.name,
+ }
+ )
+ else:
+ include_text.add(link_dicts[link_type]["including_file"])
+ sensor_items.append(link_dicts[link_type]["string_api"](transform))
+
+ render_meta_data["isolated_sensors_includes"] = list(include_text)
+ render_meta_data["isolated_sensors"] = sensor_items
+
+ rendered = base_template.render(render_meta_data)
+ print(rendered)
+
+ print("=====================================")
+ # Save the rendered template
+ with open(os.path.join(output_directory, "sensors.xacro"), "w") as file:
+ file.write(rendered)
+
+ # Write Sensor Units into separate files
+ with open(os.path.join(template_directory, "sensor_unit.xacro.template"), "r") as file:
+ sensor_units_template = Template(file.read())
+
+ for i, sensor_unit in enumerate(render_meta_data["sensor_units"]):
+ sensor_unit_calib_path = os.path.join(
+ calibration_directory, f"{sensor_unit['name']}_calibration.yaml"
+ )
+ sensor_unit_calib_yaml = load_yaml(sensor_unit_calib_path)
+ sensor_unit_calib = Calibration(sensor_unit_calib_yaml)
+ sensor_unit_render_meta_data = {}
+ sensor_unit_render_meta_data["unit_macro_name"] = sensor_unit["macro_name"]
+ sensor_unit_render_meta_data["default_config_path"] = render_meta_data[
+ "default_config_path"
+ ]
+ sensor_unit_render_meta_data["joint_unit_name"] = sensor_unit["name"]
+ sensor_unit_render_meta_data["current_base_link"] = sensor_unit_calib.base_frame
+ sensor_unit_isolated_sensors = []
+ for _, transform in sensor_unit_calib.transforms.items():
+ link_type: LinkType = obtain_link_type(transform)
+ include_text.add(link_dicts[link_type]["including_file"])
+ print(transform.name)
+ sensor_unit_isolated_sensors.append(link_dicts[link_type]["string_api"](transform))
+ sensor_unit_render_meta_data["isolated_sensors_includes"] = list(include_text)
+ sensor_unit_render_meta_data["isolated_sensors"] = sensor_unit_isolated_sensors
+
+ rendered = sensor_units_template.render(sensor_unit_render_meta_data)
+ print(rendered)
+ with open(os.path.join(output_directory, f'{sensor_unit["name"]}.xacro'), "w") as file:
+ file.write(rendered)
+ print("=====================================")
+
+ return 0
+
+
+if __name__ == "__main__":
+ import argparse
+
+ parser = argparse.ArgumentParser(description="Process four positional arguments.")
+
+ # Add four positional arguments
+ parser.add_argument("template_directory", type=str, help="The first argument")
+ parser.add_argument("calibration_directory", type=str, help="The second argument")
+ parser.add_argument("output_directory", type=str, help="The third argument")
+ parser.add_argument("project_name", type=str, help="The fourth argument")
+
+ # Parse the arguments
+ args = parser.parse_args()
+
+ main(
+ args.template_directory,
+ args.calibration_directory,
+ args.output_directory,
+ args.project_name,
+ )
diff --git a/aip_urdf_compiler/templates/sensor_unit.xacro.template b/aip_urdf_compiler/templates/sensor_unit.xacro.template
new file mode 100644
index 00000000..f0b44710
--- /dev/null
+++ b/aip_urdf_compiler/templates/sensor_unit.xacro.template
@@ -0,0 +1,31 @@
+
+
+
+ {% for item in isolated_sensors_includes %}
+
+ {% endfor %}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {% for item in isolated_sensors %}
+ {{ item }}
+ {% endfor %}
+
+
+
diff --git a/aip_urdf_compiler/templates/sensors.xacro.template b/aip_urdf_compiler/templates/sensors.xacro.template
new file mode 100644
index 00000000..318e48bd
--- /dev/null
+++ b/aip_urdf_compiler/templates/sensors.xacro.template
@@ -0,0 +1,31 @@
+
+
+
+
+
+
+ {% for item in sensor_units_includes %}
+
+ {% endfor %}
+ {% for item in isolated_sensors_includes %}
+
+ {% endfor %}
+
+ {% for item in sensor_units %}
+
+
+ {% endfor %}
+
+ {% for item in isolated_sensors %}
+
+ {{ item }}
+ {% endfor %}
+
diff --git a/aip_urdf_compiler/tests/xml_diff.py b/aip_urdf_compiler/tests/xml_diff.py
new file mode 100644
index 00000000..8dd92a34
--- /dev/null
+++ b/aip_urdf_compiler/tests/xml_diff.py
@@ -0,0 +1,274 @@
+#!/usr/bin/python3
+"""
+XML/Xacro Difference Analyzer.
+
+This script analyzes and compares two XML/Xacro files, specifically designed for ROS URDF/Xacro files
+containing sensor configurations. It identifies and reports differences in included files, sensor configurations,
+and parameter changes between two versions of a file.
+
+The analyzer is particularly useful for:
+- Tracking changes in sensor configurations
+- Validating URDF/Xacro file modifications
+- Documenting sensor setup modifications
+- Quality assurance of robot configuration changes
+
+Author: [Your Name]
+Date: [Current Date]
+"""
+
+import argparse
+from collections import OrderedDict
+from typing import Dict
+from typing import List
+import xml.etree.ElementTree as ET
+
+
+class XacroAnalyzer:
+ """
+ A class to analyze differences between two Xacro/XML files.
+
+ This class provides functionality to compare two Xacro files and identify changes
+ in includes, sensor configurations, and parameters.
+
+ Attributes:
+ original_text (str): Content of the original file
+ new_text (str): Content of the new file
+ original_root (ET.Element): XML tree root of the original file
+ new_root (ET.Element): XML tree root of the new file
+ ns (dict): Namespace dictionary for Xacro XML parsing
+ """
+
+ def __init__(self, original_file: str, new_file: str):
+ """
+ Initialize the XacroAnalyzer with two files to compare.
+
+ Args:
+ original_file (str): Path to the original Xacro file
+ new_file (str): Path to the new Xacro file
+ """
+ self.original_text = self._read_file(original_file)
+ self.new_text = self._read_file(new_file)
+ self.original_root = ET.fromstring(self.original_text)
+ self.new_root = ET.fromstring(self.new_text)
+ self.ns = {"xacro": "http://ros.org/wiki/xacro"}
+
+ @staticmethod
+ def _read_file(filename: str) -> str:
+ """
+ Read content from a file.
+
+ Args:
+ filename (str): Path to the file to read
+
+ Returns:
+ str: Content of the file
+ """
+ with open(filename, "r", encoding="utf-8") as f:
+ return f.read()
+
+ def _get_includes(self, root: ET.Element) -> List[str]:
+ """
+ Extract all include statements from an XML root.
+
+ Args:
+ root (ET.Element): XML root element to search
+
+ Returns:
+ List[str]: Sorted list of included filenames
+ """
+ includes = []
+ for include in root.findall(".//xacro:include", self.ns):
+ includes.append(include.get("filename"))
+ return sorted(includes)
+
+ def _sort_params(self, params: Dict) -> OrderedDict:
+ """
+ Sort parameters in a standardized order with common parameters first.
+
+ Args:
+ params (Dict): Dictionary of parameters to sort
+
+ Returns:
+ OrderedDict: Sorted parameters with common parameters first
+ """
+ common_params = ["name", "parent", "x", "y", "z", "roll", "pitch", "yaw"]
+ sorted_params = OrderedDict()
+
+ # Add common params in specific order
+ for param in common_params:
+ if param in params:
+ sorted_params[param] = params[param]
+
+ # Add remaining params alphabetically
+ for param in sorted(params.keys()):
+ if param not in common_params:
+ sorted_params[param] = params[param]
+
+ return sorted_params
+
+ def _get_sensors(self, root: ET.Element) -> Dict[str, List[Dict]]:
+ """
+ Extract all sensor configurations from an XML root.
+
+ Args:
+ root (ET.Element): XML root element to search
+
+ Returns:
+ Dict[str, List[Dict]]: Dictionary of sensor types mapping to their configurations
+ """
+ sensors = {"cameras": [], "lidars": [], "imus": [], "radars": []}
+
+ sensor_patterns = {
+ "cameras": ["camera", "monocular"],
+ "lidars": ["hesai", "velodyne", "pandar", "lidar"],
+ "imus": ["imu", "gnss"],
+ "radars": ["radar"],
+ }
+
+ for macro in root.findall(".//*[@name]"):
+ name = macro.get("name", "")
+ params = dict(macro.attrib)
+
+ sensor_type = None
+ for type_name, patterns in sensor_patterns.items():
+ if any(pattern in name.lower() for pattern in patterns):
+ sensor_type = type_name
+ break
+
+ if sensor_type:
+ sensors[sensor_type].append({"name": name, "params": self._sort_params(params)})
+
+ for sensor_type in sensors:
+ sensors[sensor_type].sort(key=lambda x: x["name"])
+ return sensors
+
+ def _format_params(self, params: OrderedDict) -> str:
+ """
+ Format parameters for readable output.
+
+ Args:
+ params (OrderedDict): Parameters to format
+
+ Returns:
+ str: Formatted parameter string
+ """
+ return "\n".join([f' {k}="{v}"' for k, v in params.items()])
+
+ def analyze_differences(self) -> str:
+ """
+ Analyze and report differences between the two Xacro files.
+
+ This method performs a comprehensive comparison of:
+ - Included files
+ - Sensor configurations
+ - Parameter changes
+
+ Returns:
+ str: Formatted report of all differences found
+ """
+ output = []
+
+ # 1. Compare includes
+ output.append("# Key Differences\n")
+ output.append("## 1. Include Files")
+ original_includes = self._get_includes(self.original_root)
+ new_includes = self._get_includes(self.new_root)
+
+ added_includes = set(new_includes) - set(original_includes)
+ removed_includes = set(original_includes) - set(new_includes)
+
+ if added_includes or removed_includes:
+ output.append("### Changes:")
+ if added_includes:
+ output.append("**Added:**")
+ for inc in sorted(added_includes):
+ output.append(f"- {inc}")
+ if removed_includes:
+ output.append("**Removed:**")
+ for inc in sorted(removed_includes):
+ output.append(f"- {inc}")
+ output.append("")
+
+ # 2. Compare sensors
+ original_sensors = self._get_sensors(self.original_root)
+ new_sensors = self._get_sensors(self.new_root)
+
+ output.append("## 2. Sensor Configuration Changes")
+
+ for sensor_type in ["cameras", "lidars", "imus", "radars"]:
+ orig_sensor = original_sensors[sensor_type]
+ new_sensor = new_sensors[sensor_type]
+
+ if orig_sensor or new_sensor:
+ output.append(f"\n### {sensor_type.title()}")
+
+ # Compare sensor names
+ orig_names = [s["name"] for s in orig_sensor]
+ new_names = [s["name"] for s in new_sensor]
+
+ if orig_names != new_names:
+ output.append("#### Name Changes:")
+ output.append(
+ f"**Original ({len(orig_names)})**: " + ", ".join(sorted(orig_names))
+ )
+ output.append(f"**New ({len(new_names)})**: " + ", ".join(sorted(new_names)))
+
+ # Compare parameters
+ if orig_sensor and new_sensor:
+ output.append("\n#### Parameter Changes:")
+
+ # Compare parameters of first sensor of each type
+ orig_params = orig_sensor[0]["params"]
+ new_params = new_sensor[0]["params"]
+
+ added_params = set(new_params.keys()) - set(orig_params.keys())
+ removed_params = set(orig_params.keys()) - set(new_params.keys())
+ changed_params = {
+ k: (orig_params[k], new_params[k])
+ for k in set(orig_params.keys()) & set(new_params.keys())
+ if orig_params[k] != new_params[k]
+ }
+
+ if added_params:
+ output.append("**Added parameters:**")
+ for param in sorted(added_params):
+ output.append(f'- {param}: "{new_params[param]}"')
+
+ if removed_params:
+ output.append("**Removed parameters:**")
+ for param in sorted(removed_params):
+ output.append(f'- {param}: "{orig_params[param]}"')
+
+ if changed_params:
+ output.append("**Modified parameters:**")
+ for param, (old_val, new_val) in sorted(changed_params.items()):
+ output.append(f'- {param}: "{old_val}" → "{new_val}"')
+
+ return "\n".join(output)
+
+
+def main():
+ # Create argument parser
+ parser = argparse.ArgumentParser(
+ description="Compare two XACRO files and analyze their differences"
+ )
+
+ # Add arguments
+ parser.add_argument(
+ "--original", "-o", required=True, help="Path to the original sensors.xacro file"
+ )
+
+ parser.add_argument("--new", "-n", required=True, help="Path to the new sensors.xacro file")
+
+ # Parse arguments
+ args = parser.parse_args()
+
+ # Create analyzer instance with provided file paths
+ analyzer = XacroAnalyzer(args.original, args.new)
+
+ # Print analysis results
+ print(analyzer.analyze_differences())
+
+
+if __name__ == "__main__":
+ main()
diff --git a/aip_xx1_description/CMakeLists.txt b/aip_xx1_description/CMakeLists.txt
index 50723262..362fb6da 100644
--- a/aip_xx1_description/CMakeLists.txt
+++ b/aip_xx1_description/CMakeLists.txt
@@ -2,9 +2,12 @@ cmake_minimum_required(VERSION 3.5)
project(aip_xx1_description)
find_package(ament_cmake_auto REQUIRED)
+find_package(aip_urdf_compiler REQUIRED)
ament_auto_find_build_dependencies()
+aip_cmake_urdf_compile()
+
ament_auto_package(INSTALL_TO_SHARE
urdf
config
diff --git a/aip_xx1_description/config/sensor_kit_calibration.yaml b/aip_xx1_description/config/sensor_kit_calibration.yaml
index ab417b52..69c94144 100644
--- a/aip_xx1_description/config/sensor_kit_calibration.yaml
+++ b/aip_xx1_description/config/sensor_kit_calibration.yaml
@@ -6,6 +6,7 @@ sensor_kit_base_link:
roll: -0.025
pitch: 0.315
yaw: 1.035
+ type: monocular_camera
camera1/camera_link:
x: -0.10731
y: -0.56343
@@ -13,6 +14,7 @@ sensor_kit_base_link:
roll: -0.025
pitch: 0.32
yaw: -2.12
+ type: monocular_camera
camera2/camera_link:
x: 0.10731
y: -0.56343
@@ -20,6 +22,7 @@ sensor_kit_base_link:
roll: -0.00
pitch: 0.335
yaw: -1.04
+ type: monocular_camera
camera3/camera_link:
x: -0.10731
y: 0.56343
@@ -27,6 +30,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: 0.325
yaw: 2.0943951
+ type: monocular_camera
camera4/camera_link:
x: 0.07356
y: 0.0
@@ -34,6 +38,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: -0.03
yaw: -0.005
+ type: monocular_camera
camera5/camera_link:
x: -0.07356
y: 0.0
@@ -41,6 +46,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: -0.01
yaw: 3.125
+ type: monocular_camera
camera6/camera_link:
x: 0.05
y: 0.0175
@@ -48,6 +54,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: 0.0
yaw: 0.0
+ type: monocular_camera
camera7/camera_link:
x: 0.05
y: -0.0175
@@ -55,6 +62,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: 0.0
yaw: 0.0
+ type: monocular_camera
velodyne_top_base_link:
x: 0.0
y: 0.0
@@ -62,6 +70,7 @@ sensor_kit_base_link:
roll: 0.0
pitch: 0.0
yaw: 1.575
+ type: VLS-128.urdf
velodyne_left_base_link:
x: 0.0
y: 0.56362
@@ -69,6 +78,7 @@ sensor_kit_base_link:
roll: -0.02
pitch: 0.71
yaw: 1.575
+ type: VLP-16.urdf
velodyne_right_base_link:
x: 0.0
y: -0.56362
@@ -76,6 +86,7 @@ sensor_kit_base_link:
roll: -0.01
pitch: 0.71
yaw: -1.580
+ type: VLP-16.urdf
gnss_link:
x: -0.1
y: 0.0
@@ -83,6 +94,8 @@ sensor_kit_base_link:
roll: 0.0
pitch: 0.0
yaw: 0.0
+ type: imu
+ frame_id: gnss
tamagawa/imu_link:
x: 0.0
y: 0.0
@@ -90,3 +103,5 @@ sensor_kit_base_link:
roll: 3.14159265359
pitch: 0.0
yaw: 3.14159265359
+ type: imu
+ frame_id: tamagawa/imu
diff --git a/aip_xx1_description/config/sensors_calibration.yaml b/aip_xx1_description/config/sensors_calibration.yaml
index e8c4b75e..c7de4df6 100644
--- a/aip_xx1_description/config/sensors_calibration.yaml
+++ b/aip_xx1_description/config/sensors_calibration.yaml
@@ -6,6 +6,7 @@ base_link:
roll: 0.0
pitch: 0.0
yaw: 0.0
+ type: radar
sensor_kit_base_link:
x: 0.9
y: 0.0
@@ -13,6 +14,7 @@ base_link:
roll: -0.001
pitch: 0.015
yaw: -0.0364
+ type: units
livox_front_right_base_link:
x: 3.290
y: -0.65485
@@ -20,6 +22,7 @@ base_link:
roll: 0.0
pitch: 0.0
yaw: -0.872664444
+ type: livox_horizon
livox_front_left_base_link:
x: 3.290
y: 0.65485
@@ -27,6 +30,7 @@ base_link:
roll: -0.021
pitch: 0.05
yaw: 0.872664444
+ type: livox_horizon
velodyne_rear_base_link:
x: -0.358
y: 0.0
@@ -34,3 +38,4 @@ base_link:
roll: -0.02
pitch: 0.7281317
yaw: 3.141592
+ type: VLP-16.urdf
diff --git a/aip_xx1_description/package.xml b/aip_xx1_description/package.xml
index b35b48e0..73954095 100644
--- a/aip_xx1_description/package.xml
+++ b/aip_xx1_description/package.xml
@@ -7,8 +7,10 @@
Yukihiro Saito
Apache 2
+ ament_cmake
ament_cmake_auto
+ aip_urdf_compiler
velodyne_description
diff --git a/aip_xx1_description/urdf/.gitignore b/aip_xx1_description/urdf/.gitignore
new file mode 100644
index 00000000..e1e98315
--- /dev/null
+++ b/aip_xx1_description/urdf/.gitignore
@@ -0,0 +1 @@
+*.xacro
diff --git a/aip_xx1_description/urdf/radar.xacro b/aip_xx1_description/urdf/radar.xacro
deleted file mode 100644
index 8b0f8d4b..00000000
--- a/aip_xx1_description/urdf/radar.xacro
+++ /dev/null
@@ -1,31 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/aip_xx1_description/urdf/sensor_kit.xacro b/aip_xx1_description/urdf/sensor_kit.xacro
deleted file mode 100644
index ef36763f..00000000
--- a/aip_xx1_description/urdf/sensor_kit.xacro
+++ /dev/null
@@ -1,228 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/aip_xx1_description/urdf/sensors.xacro b/aip_xx1_description/urdf/sensors.xacro
deleted file mode 100644
index 0484bdc3..00000000
--- a/aip_xx1_description/urdf/sensors.xacro
+++ /dev/null
@@ -1,72 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/aip_xx1_gen2_description/CMakeLists.txt b/aip_xx1_gen2_description/CMakeLists.txt
index 549de0f8..782df039 100644
--- a/aip_xx1_gen2_description/CMakeLists.txt
+++ b/aip_xx1_gen2_description/CMakeLists.txt
@@ -2,9 +2,12 @@ cmake_minimum_required(VERSION 3.5)
project(aip_xx1_gen2_description)
find_package(ament_cmake_auto REQUIRED)
+find_package(aip_urdf_compiler REQUIRED)
ament_auto_find_build_dependencies()
+aip_cmake_urdf_compile()
+
ament_auto_package(INSTALL_TO_SHARE
urdf
config
diff --git a/aip_xx1_gen2_description/config/sensor_kit_calibration.yaml b/aip_xx1_gen2_description/config/sensor_kit_calibration.yaml
index 88288533..d6025d6f 100644
--- a/aip_xx1_gen2_description/config/sensor_kit_calibration.yaml
+++ b/aip_xx1_gen2_description/config/sensor_kit_calibration.yaml
@@ -6,6 +6,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: monocular_camera
camera1/camera_link:
x: 0.372 # Design Value
y: 0.045 # Design Value
@@ -13,6 +14,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: monocular_camera
camera2/camera_link:
x: 0.372 # Design Value
y: -0.045 # Design Value
@@ -20,6 +22,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: monocular_camera
camera3/camera_link:
x: 0.133 # Design Value
y: 0.498 # Design Value
@@ -27,6 +30,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.872665 # Design Value
+ type: monocular_camera
camera4/camera_link:
x: 0.133 # Design Value
y: -0.498 # Design Value
@@ -34,6 +38,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -0.872665 # Design Value
+ type: monocular_camera
camera5/camera_link:
x: 0.095 # Design Value
y: 0.524 # Design Value
@@ -41,6 +46,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 1.0472 # Design Value
+ type: monocular_camera
camera6/camera_link:
x: 0.095 # Design Value
y: -0.524 # Design Value
@@ -48,6 +54,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -1.0472 # Design Value
+ type: monocular_camera
camera7/camera_link:
x: -0.345 # Design Value
y: 0.244 # Design Value
@@ -55,6 +62,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 2.70526 # Design Value
+ type: monocular_camera
camera8/camera_link:
x: -0.345 # Design Value
y: -0.244 # Design Value
@@ -62,6 +70,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -2.70526 # Design Value
+ type: monocular_camera
camera9/camera_link:
x: -0.362 # Design Value
y: 0.202 # Design Value
@@ -69,6 +78,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 2.79253 # Design Value
+ type: monocular_camera
camera10/camera_link:
x: -0.362 # Design Value
y: -0.202 # Design Value
@@ -76,6 +86,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -2.79253 # Design Value
+ type: monocular_camera
hesai_top_base_link:
x: 0.0 # Design Value
y: 0.0 # Design Value
@@ -83,6 +94,7 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 4.36332298038 # Design Value
+ type: pandar_ot128
hesai_left_base_link:
x: 0.0 # Design Value
y: 0.564 # Design Value
@@ -90,6 +102,8 @@ sensor_kit_base_link:
roll: 0.872665 # Design Value
pitch: 0.0 # Design Value
yaw: 3.14159265359 # Design Value
+ type: pandar_xt32
+ frame_id: hesai_side_left
hesai_right_base_link:
x: 0.0 # Design Value
y: -0.564 # Design Value
@@ -97,6 +111,8 @@ sensor_kit_base_link:
roll: 0.69813132679 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: pandar_xt32
+ frame_id: hesai_side_right
gnss_link:
x: -0.279 # Design Value
y: 0.0 # Design Value
@@ -104,6 +120,8 @@ sensor_kit_base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: imu
+ frame_id: gnss
tamagawa/imu_link:
x: -0.129 # Design Value
y: 0.0 # Design Value
@@ -111,3 +129,5 @@ sensor_kit_base_link:
roll: 3.14159265359
pitch: 0.0 # Design Value
yaw: 3.14159265359 # Design Value
+ type: imu
+ frame_id: tamagawa/imu
diff --git a/aip_xx1_gen2_description/config/sensors_calibration.yaml b/aip_xx1_gen2_description/config/sensors_calibration.yaml
index a57d3ea9..4fb70f70 100644
--- a/aip_xx1_gen2_description/config/sensors_calibration.yaml
+++ b/aip_xx1_gen2_description/config/sensors_calibration.yaml
@@ -6,6 +6,7 @@ base_link:
roll: 0.0
pitch: 0.0
yaw: 0.0
+ type: units
hesai_front_left_base_link:
x: 3.373 # Design Value
y: 0.740 # Design Value
@@ -13,6 +14,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 2.44346132679 # Design Value
+ type: pandar_xt32
hesai_front_right_base_link:
x: 3.373 # Design Value
y: -0.740 # Design Value
@@ -20,6 +22,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.69813132679 # Design Value
+ type: pandar_xt32
# velodyne_rear_base_link: #unused
# x: -0.358
# y: 0.0
@@ -34,6 +37,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 0.0 # Design Value
+ type: radar
front_right/radar_link:
x: 3.384 # Design Value
y: -0.7775 # Design Value
@@ -41,6 +45,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -1.22173 # Design Value
+ type: radar
front_left/radar_link:
x: 3.384 # Design Value
y: 0.7775 # Design Value
@@ -48,6 +53,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 1.22173 # Design Value
+ type: radar
rear_center/radar_link:
x: -0.858 # Design Value
y: 0.0 # Design Value
@@ -55,6 +61,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 3.141592 # Design Value
+ type: radar
rear_right/radar_link:
x: -0.782 # Design Value
y: -0.761 # Design Value
@@ -62,6 +69,7 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: -2.0944 # Design Value
+ type: radar
rear_left/radar_link:
x: -0.782 # Design Value
y: 0.761 # Design Value
@@ -69,3 +77,4 @@ base_link:
roll: 0.0 # Design Value
pitch: 0.0 # Design Value
yaw: 2.0944 # Design Value
+ type: radar
diff --git a/aip_xx1_gen2_description/package.xml b/aip_xx1_gen2_description/package.xml
index 9b010d72..e1f2774f 100644
--- a/aip_xx1_gen2_description/package.xml
+++ b/aip_xx1_gen2_description/package.xml
@@ -9,6 +9,7 @@
ament_cmake_auto
+ aip_urdf_compiler
velodyne_description
diff --git a/aip_xx1_gen2_description/urdf/.gitignore b/aip_xx1_gen2_description/urdf/.gitignore
new file mode 100644
index 00000000..e1e98315
--- /dev/null
+++ b/aip_xx1_gen2_description/urdf/.gitignore
@@ -0,0 +1 @@
+*.xacro
diff --git a/aip_xx1_gen2_description/urdf/sensor_kit.xacro b/aip_xx1_gen2_description/urdf/sensor_kit.xacro
deleted file mode 100644
index 137b3589..00000000
--- a/aip_xx1_gen2_description/urdf/sensor_kit.xacro
+++ /dev/null
@@ -1,253 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/aip_xx1_gen2_description/urdf/sensors.xacro b/aip_xx1_gen2_description/urdf/sensors.xacro
deleted file mode 100644
index 79c2c15f..00000000
--- a/aip_xx1_gen2_description/urdf/sensors.xacro
+++ /dev/null
@@ -1,103 +0,0 @@
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-