From 82007da6c10676a912fa8077efc2269b45e05efb Mon Sep 17 00:00:00 2001
From: Simon Wittl <simon.wittl@th-deg.de>
Date: Thu, 12 Dec 2024 07:21:44 +0100
Subject: [PATCH] Thd io

---
 example/convert_projections.py                |  25 ++
 example/projection_example.py                 |  10 +-
 example/tigre_example.py                      |  42 +--
 example/volume_example.py                     |  17 +-
 rq_controller/common/__init__.py              |   4 +-
 rq_controller/common/io/loader.py             |  24 +-
 rq_controller/common/io/rq_json/__init__.py   |   4 +-
 rq_controller/common/io/rq_json/json_load.py  |  90 ++++--
 rq_controller/common/io/rq_json/json_write.py | 101 ++++--
 rq_controller/common/io/thd/__init__.py       |   5 +
 rq_controller/common/io/thd/thd_load.py       | 125 +++++++
 rq_controller/common/io/thd/thd_write.py      |  86 +++++
 rq_controller/common/io/writer.py             | 107 +++---
 rq_controller/common/projection.py            | 304 ++++++++++++------
 rq_controller/common/projection_geometry.py   | 116 +++++--
 rq_controller/common/region_of_intrest.py     |  80 ++---
 rq_controller/common/volume.py                |  91 +++---
 rq_controller/rq_workflow.py                  |  61 ++--
 rq_controller/tf2/static_broadcaster.py       |  42 +--
 setup.py                                      |  29 +-
 ...st_base_loader.cpython-38-pytest-8.2.2.pyc | Bin 2839 -> 2839 bytes
 ...st_base_writer.cpython-38-pytest-8.2.2.pyc | Bin 10618 -> 10630 bytes
 ...est_projection.cpython-38-pytest-8.2.2.pyc | Bin 20692 -> 19665 bytes
 ...ction_geometry.cpython-38-pytest-8.2.2.pyc | Bin 11871 -> 11883 bytes
 ...ion_of_intrest.cpython-38-pytest-8.2.2.pyc | Bin 12566 -> 12566 bytes
 .../test_volume.cpython-38-pytest-8.2.2.pyc   | Bin 9180 -> 9180 bytes
 test/test_base_loader.py                      |  10 +-
 test/test_base_writer.py                      |  55 +++-
 test/test_projection.py                       |  73 ++++-
 test/test_projection_geometry.py              |  54 ++--
 test/test_region_of_intrest.py                |  31 +-
 test/test_volume.py                           |  24 +-
 32 files changed, 1118 insertions(+), 492 deletions(-)
 create mode 100644 example/convert_projections.py
 create mode 100644 rq_controller/common/io/thd/__init__.py
 create mode 100644 rq_controller/common/io/thd/thd_load.py
 create mode 100644 rq_controller/common/io/thd/thd_write.py

diff --git a/example/convert_projections.py b/example/convert_projections.py
new file mode 100644
index 0000000..b5f5051
--- /dev/null
+++ b/example/convert_projections.py
@@ -0,0 +1,25 @@
+from rq_controller.common.io.rq_json import RqJsonWriter
+from rq_controller.common.io.thd import RawLoader
+from pathlib import Path
+
+
+FOLDER = Path(r"C:\Users\swittl\Downloads\defect_detection\defect_detection\object_2")
+SAVE_FOLDER = Path(
+    r"C:\Users\swittl\Downloads\defect_detection\defect_detection\rq_converted\object_2"
+)
+
+
+def main():
+    writer = RqJsonWriter()
+    loader = RawLoader()
+
+    projections_files = FOLDER.glob("*.raw")
+
+    for i, file in enumerate(projections_files):
+        projection = loader.load_projection(file)
+        save_path = writer.get_next_projection_save_path(SAVE_FOLDER)
+        writer.write_projection(save_path, projection, optional=True)
+
+
+if __name__ == "__main__":
+    main()
diff --git a/example/projection_example.py b/example/projection_example.py
index 4b6fd05..23df07d 100644
--- a/example/projection_example.py
+++ b/example/projection_example.py
@@ -3,7 +3,8 @@ from rq_controller.common.io.rq_json import RqJsonWriter, RqJsonLoader
 from pathlib import Path
 
 
-FOLDER = Path('./example/data')
+FOLDER = Path("./example/data")
+
 
 def main():
     projection_2 = PyProjection.dummy()
@@ -13,12 +14,13 @@ def main():
     loader = RqJsonLoader()
 
     writer.write_projection(writer.get_projection_save_path_i(FOLDER, 2), projection_2)
-    
+
     projection_1 = loader.load_projection(writer.get_projection_save_path_i(FOLDER, 1))
     print(projection_1)
 
     projection_2 = loader.load_projection(writer.get_projection_save_path_i(FOLDER, 2))
     print(projection_2)
 
-if __name__ == '__main__':
-    main()
\ No newline at end of file
+
+if __name__ == "__main__":
+    main()
diff --git a/example/tigre_example.py b/example/tigre_example.py
index fd7c551..be4aaeb 100644
--- a/example/tigre_example.py
+++ b/example/tigre_example.py
@@ -5,7 +5,6 @@ import rclpy
 
 from scipy.spatial.transform import Rotation
 import numpy as np
-import matplotlib.pyplot as plt
 
 
 # !!!
@@ -13,8 +12,9 @@ import matplotlib.pyplot as plt
 # !!!
 
 NUMBER_OF_PROJECTION = 80
-FOD_MM = 1000.
-FDD_MM = 2000.
+FOD_MM = 1000.0
+FDD_MM = 2000.0
+
 
 def main():
     # Initialize workflow node
@@ -24,10 +24,10 @@ def main():
     projection_stack: list[PyProjection] = list()
     projection = PyProjection.dummy()
     projection.image = np.zeros((1000, 1000), dtype=np.uint16)
-    projection.voltage_kv = 200.
-    projection.current_ua = 10.
-    projection.detector_heigth_mm = 200.
-    projection.detector_width_mm = 200.
+    projection.voltage_kv = 200.0
+    projection.current_ua = 10.0
+    projection.detector_heigth_mm = 200.0
+    projection.detector_width_mm = 200.0
 
     # create the source and detector positions
     source = np.array([FOD_MM, 0, 0])
@@ -36,21 +36,25 @@ def main():
 
     # Move source / dtector and aquire projections
     for i in range(NUMBER_OF_PROJECTION):
-        rotation = Rotation.from_euler('Z', angles[i], False)
-        scan_pose = projection.look_at(rotation.apply(source) + (np.random.random(3) - 0.5) * 30, 
-                                       rotation.apply(detector) + (np.random.random(3) - 0.5) * 30, 
-                                       np.array([0, 0, -1]))
+        rotation = Rotation.from_euler("Z", angles[i], False)
+        scan_pose = projection.look_at(
+            rotation.apply(source) + (np.random.random(3) - 0.5) * 30,
+            rotation.apply(detector) + (np.random.random(3) - 0.5) * 30,
+            np.array([0, 0, -1]),
+        )
         projection_stack.append(workflow.aquire_projection(scan_pose))
 
     # Define reconstruction area and call reconstruction client
-    roi = PyRegionOfIntrest(center_points_mm=np.array([0., 0., 0.]),
-                            dimensions_mm=np.array([120., 120., 120.]), 
-                            resolution_mm=np.array([0.5, 0.5, 0.5]))
+    roi = PyRegionOfIntrest(
+        center_points_mm=np.array([0.0, 0.0, 0.0]),
+        dimensions_mm=np.array([120.0, 120.0, 120.0]),
+        resolution_mm=np.array([0.5, 0.5, 0.5]),
+    )
+
+    workflow.reconstruction.set_reconstruction_algorithm_name("ossart")  # ossart / fdk
+    workflow.get_volume(projection_stack, roi)
 
-    workflow.reconstruction.set_reconstruction_algorithm_name('ossart') # ossart / fdk
-    volume = workflow.get_volume(projection_stack, roi)
 
-    
-if __name__ == '__main__':
+if __name__ == "__main__":
     rclpy.init()
-    main()
\ No newline at end of file
+    main()
diff --git a/example/volume_example.py b/example/volume_example.py
index b462199..68d515c 100644
--- a/example/volume_example.py
+++ b/example/volume_example.py
@@ -4,34 +4,33 @@ from pathlib import Path
 import numpy as np
 
 
-FOLDER = Path('./example/data')
+FOLDER = Path("./example/data")
 
 
 def main():
     volume = PyVolume.dummy()
-    print(f'Shape (x / y / z): {volume.shape}')
+    print(f"Shape (x / y / z): {volume.shape}")
     msg = volume.as_message()
-    print(f'Number of slices: {len(msg.slices)}')
+    print(f"Number of slices: {len(msg.slices)}")
 
     volume_2 = PyVolume.from_message(msg)
-    print(f'Shape (x / y / z): {volume_2.shape}')
+    print(f"Shape (x / y / z): {volume_2.shape}")
 
     writer = RqJsonWriter()
     writer.write_volume(writer.get_volume_save_path_i(FOLDER, 1), volume_2)
 
     loader = RqJsonLoader()
     volume_3 = loader.load_volume(writer.get_volume_save_path_i(FOLDER, 1))
-    print(f'Shape (x / y / z): {volume_3.shape}')
+    print(f"Shape (x / y / z): {volume_3.shape}")
 
     grid = volume_3.roi.get_grid()
-    print(f'grid shape: {grid.shape}')
+    print(f"grid shape: {grid.shape}")
 
-    pos = volume_3.roi.next_neighbor(grid, np.array([0., 0., 0.]))
+    pos = volume_3.roi.next_neighbor(grid, np.array([0.0, 0.0, 0.0]))
     print(pos)
     value = volume_3.array[pos[0], pos[1], pos[2]]
     print(value)
 
 
-if __name__ == '__main__':
+if __name__ == "__main__":
     main()
-
diff --git a/rq_controller/common/__init__.py b/rq_controller/common/__init__.py
index 49190b8..c0676a1 100644
--- a/rq_controller/common/__init__.py
+++ b/rq_controller/common/__init__.py
@@ -1,2 +1,4 @@
 from .projection import PyProjectionGeometry, PyProjection
-from .volume import PyVolume, PyRegionOfIntrest
\ No newline at end of file
+from .volume import PyVolume, PyRegionOfIntrest
+
+__all__ = ["PyProjectionGeometry", "PyProjection", "PyVolume", "PyRegionOfIntrest"]
diff --git a/rq_controller/common/io/loader.py b/rq_controller/common/io/loader.py
index 4ce8e32..66c7b5c 100644
--- a/rq_controller/common/io/loader.py
+++ b/rq_controller/common/io/loader.py
@@ -1,16 +1,15 @@
-import numpy as np
-import json
 from pathlib import Path
 from ...common import PyProjectionGeometry, PyProjection, PyRegionOfIntrest, PyVolume
 
 
-class BaseDataLoader():
-    def __init__(self, 
-                 porjection_geometry_suffix: str, 
-                 projection_suffix: str,
-                 region_of_intrest_suffix: str,
-                 volume_suffix: str):
-        
+class BaseDataLoader:
+    def __init__(
+        self,
+        porjection_geometry_suffix: str,
+        projection_suffix: str,
+        region_of_intrest_suffix: str,
+        volume_suffix: str,
+    ):
         self.porjection_geometry_suffix = porjection_geometry_suffix
         self.projection_suffix = projection_suffix
         self.region_of_intrest_suffix = region_of_intrest_suffix
@@ -18,13 +17,12 @@ class BaseDataLoader():
 
     def load_projection_geometry(self, load_path: Path) -> PyProjectionGeometry:
         raise NotImplementedError()
-    
+
     def load_projection(self, load_path: Path) -> PyProjection:
         raise NotImplementedError()
-    
+
     def load_region_of_intrest(self, load_path: Path) -> PyRegionOfIntrest:
         raise NotImplementedError()
-    
+
     def load_volume(self, load_path: Path) -> PyVolume:
         raise NotImplementedError()
-    
\ No newline at end of file
diff --git a/rq_controller/common/io/rq_json/__init__.py b/rq_controller/common/io/rq_json/__init__.py
index f6f5812..4d08736 100644
--- a/rq_controller/common/io/rq_json/__init__.py
+++ b/rq_controller/common/io/rq_json/__init__.py
@@ -1,2 +1,4 @@
 from .json_load import RqJsonLoader
-from .json_write import RqJsonWriter
\ No newline at end of file
+from .json_write import RqJsonWriter
+
+__all__ = ["RqJsonLoader", "RqJsonWriter"]
diff --git a/rq_controller/common/io/rq_json/json_load.py b/rq_controller/common/io/rq_json/json_load.py
index 2a9d6e6..0bd1a5d 100644
--- a/rq_controller/common/io/rq_json/json_load.py
+++ b/rq_controller/common/io/rq_json/json_load.py
@@ -1,18 +1,24 @@
 import numpy as np
-import json 
+import json
 from pathlib import Path
 
-from ..loader import BaseDataLoader, PyProjection, PyProjectionGeometry, PyRegionOfIntrest, PyVolume
+from ..loader import (
+    BaseDataLoader,
+    PyProjection,
+    PyProjectionGeometry,
+    PyRegionOfIntrest,
+    PyVolume,
+)
 from PIL import Image
 import pyometiff
 
 
 class RqJsonLoader(BaseDataLoader):
     def __init__(self):
-        super().__init__('.geom-json', '.tif', '.roi-json', '.ome.tiff')
+        super().__init__(".geom-json", ".tif", ".roi-json", ".ome.tiff")
 
     def load_json(self, load_path: Path) -> dict:
-        with open(str(load_path), 'r') as f:
+        with open(str(load_path), "r") as f:
             data_dict = json.load(f)
         return data_dict
 
@@ -20,57 +26,71 @@ class RqJsonLoader(BaseDataLoader):
         data_dict = self.load_json(load_path)
 
         projection_geometry = PyProjectionGeometry(
-            focal_spot_mm=np.array([data_dict['focal_spot_mm']]),
-            detector_postion_mm=np.array([data_dict['detector_postion_mm']]),
-            detector_orientation_quad=np.array([data_dict['detector_orientation_quad']]),
-            frame_id=data_dict['frame_id'],
-            focal_spot_orientation_quad=np.array(data_dict['focal_spot_orientation_quad']))
-        
+            focal_spot_mm=np.array([data_dict["focal_spot_mm"]]),
+            detector_postion_mm=np.array([data_dict["detector_postion_mm"]]),
+            detector_orientation_quad=np.array(
+                [data_dict["detector_orientation_quad"]]
+            ),
+            frame_id=data_dict["frame_id"],
+            focal_spot_orientation_quad=np.array(
+                data_dict["focal_spot_orientation_quad"]
+            ),
+        )
+
         return projection_geometry
-    
+
     def load_projection(self, load_path: Path) -> PyProjection:
-        load_path_projection_geometry = load_path.parent / f'{load_path.stem}{self.porjection_geometry_suffix}'
+        load_path_projection_geometry = (
+            load_path.parent / f"{load_path.stem}{self.porjection_geometry_suffix}"
+        )
         data_dict = self.load_json(load_path_projection_geometry)
         image = np.asarray(Image.open(load_path))
 
         projection = PyProjection(
-            focal_spot_mm=np.array([data_dict['focal_spot_mm']]),
-            detector_postion_mm=np.array([data_dict['detector_postion_mm']]),
-            detector_orientation_quad=np.array([data_dict['detector_orientation_quad']]),
+            focal_spot_mm=np.array([data_dict["focal_spot_mm"]]),
+            detector_postion_mm=np.array([data_dict["detector_postion_mm"]]),
+            detector_orientation_quad=np.array(
+                [data_dict["detector_orientation_quad"]]
+            ),
             image=image,
-            detector_heigth_mm=data_dict['detector_heigth_mm'],
-            detector_width_mm=data_dict['detector_width_mm'],
-            frame_id=data_dict['frame_id'],
-            focal_spot_orientation_quad=np.array(data_dict['focal_spot_orientation_quad']),
-            voltage_kv=data_dict['voltage_kv'],
-            current_ua=data_dict['current_ua'],
-            exposure_time_ms=data_dict['exposure_time_ms'])
-        
+            detector_heigth_mm=data_dict["detector_heigth_mm"],
+            detector_width_mm=data_dict["detector_width_mm"],
+            frame_id=data_dict["frame_id"],
+            focal_spot_orientation_quad=np.array(
+                data_dict["focal_spot_orientation_quad"]
+            ),
+            voltage_kv=data_dict["voltage_kv"],
+            current_ua=data_dict["current_ua"],
+            exposure_time_ms=data_dict["exposure_time_ms"],
+        )
+
         return projection
-    
+
     def load_region_of_intrest(self, load_path: Path) -> PyRegionOfIntrest:
         data_dict = self.load_json(load_path)
 
         region_of_intrest = PyRegionOfIntrest(
-            center_points_mm=np.array(data_dict['center_points_mm']),
-            dimensions_mm=np.array(data_dict['dimensions_mm']),
-            frame_id=data_dict['frame_id'],
-            resolution_mm=np.array(data_dict['resolution_mm']))
-        
+            center_points_mm=np.array(data_dict["center_points_mm"]),
+            dimensions_mm=np.array(data_dict["dimensions_mm"]),
+            frame_id=data_dict["frame_id"],
+            resolution_mm=np.array(data_dict["resolution_mm"]),
+        )
+
         return region_of_intrest
-    
+
     def load_volume(self, load_path: Path) -> PyVolume:
-        load_path_roi = load_path.parent / f'{load_path.stem}{self.region_of_intrest_suffix}'
+        load_path_roi = (
+            load_path.parent / f"{load_path.stem}{self.region_of_intrest_suffix}"
+        )
         roi = self.load_region_of_intrest(load_path_roi)
         reader = pyometiff.OMETIFFReader(load_path)
         volume, _, _ = reader.read()
-        
+
         if volume.dtype == np.uint16:
             data_type = 0
         elif volume.dtype == np.uint8:
             data_type = 1
         else:
-            raise ValueError('data type not implemented.')
-        
-        return PyVolume(volume, roi, data_type)
+            raise ValueError("data type not implemented.")
 
+        return PyVolume(volume, roi, data_type)
diff --git a/rq_controller/common/io/rq_json/json_write.py b/rq_controller/common/io/rq_json/json_write.py
index e5d6de8..0d14c0d 100644
--- a/rq_controller/common/io/rq_json/json_write.py
+++ b/rq_controller/common/io/rq_json/json_write.py
@@ -1,73 +1,104 @@
-import numpy as np
-import json 
+import json
 from pathlib import Path
 
-from ..writer import BaseDataWriter, PyProjection, PyProjectionGeometry, PyRegionOfIntrest, PyVolume
+from ..writer import (
+    BaseDataWriter,
+    PyProjection,
+    PyProjectionGeometry,
+    PyRegionOfIntrest,
+    PyVolume,
+)
 from PIL import Image
 import pyometiff
 
 
 class RqJsonWriter(BaseDataWriter):
     def __init__(self):
-        super().__init__('.geom-json', '.tif', '.roi-json', '.ome.tiff')
+        super().__init__(".geom-json", ".tif", ".roi-json", ".ome.tiff")
 
     def write_json(self, save_path: Path, data_dict: dict):
-        with open(str(save_path), 'w') as f:
+        with open(str(save_path), "w") as f:
             json.dump(data_dict, f, indent=2)
 
-    def write_projection_geometry(self, save_path: Path, projection_geometry: PyProjectionGeometry):
+    def write_projection_geometry(
+        self, save_path: Path, projection_geometry: PyProjectionGeometry
+    ):
         data_dict = dict()
 
-        #geometry
-        data_dict['focal_spot_mm'] = projection_geometry.focal_spot_mm.tolist()
-        data_dict['detector_postion_mm'] = projection_geometry.detector_postion_mm.tolist()
-        data_dict['detector_orientation_quad'] = projection_geometry.detector_orientation_quad.tolist()
-        data_dict['focal_spot_orientation_quad'] = projection_geometry.focal_spot_orientation_quad.tolist()
-        
-        data_dict['frame_id'] = projection_geometry.frame_id
+        # geometry
+        data_dict["focal_spot_mm"] = projection_geometry.focal_spot_mm.tolist()
+        data_dict["detector_postion_mm"] = (
+            projection_geometry.detector_postion_mm.tolist()
+        )
+        data_dict["detector_orientation_quad"] = (
+            projection_geometry.detector_orientation_quad.tolist()
+        )
+        data_dict["focal_spot_orientation_quad"] = (
+            projection_geometry.focal_spot_orientation_quad.tolist()
+        )
+
+        data_dict["frame_id"] = projection_geometry.frame_id
 
         self.write_json(save_path, data_dict)
 
-    def write_projection(self, save_path: Path, projection: PyProjection):
-        save_path_projection_geometry = save_path.parent / f'{save_path.stem}{self.porjection_geometry_suffix}'
+    def write_projection(
+        self, save_path: Path, projection: PyProjection, optional: bool = False
+    ):
+        save_path_projection_geometry = (
+            save_path.parent / f"{save_path.stem}{self.porjection_geometry_suffix}"
+        )
         data_dict = dict()
 
         # geometry
-        data_dict['focal_spot_mm'] = projection.focal_spot_mm.tolist()
-        data_dict['detector_postion_mm'] = projection.detector_postion_mm.tolist()
-        data_dict['detector_orientation_quad'] = projection.detector_orientation_quad.tolist()
-        data_dict['focal_spot_orientation_quad'] = projection.focal_spot_orientation_quad.tolist()
-        
+        data_dict["focal_spot_mm"] = projection.focal_spot_mm.tolist()
+        data_dict["detector_postion_mm"] = projection.detector_postion_mm.tolist()
+        data_dict["detector_orientation_quad"] = (
+            projection.detector_orientation_quad.tolist()
+        )
+        data_dict["focal_spot_orientation_quad"] = (
+            projection.focal_spot_orientation_quad.tolist()
+        )
+
         # detector
-        data_dict['pixel_pitch_x_mm'] = projection.pixel_pitch_x_mm
-        data_dict['pixel_pitch_y_mm'] = projection.pixel_pitch_y_mm
-        data_dict['detector_heigth_mm'] = projection.detector_heigth_mm
-        data_dict['detector_width_mm'] = projection.detector_width_mm
+        data_dict["pixel_pitch_x_mm"] = projection.pixel_pitch_x_mm
+        data_dict["pixel_pitch_y_mm"] = projection.pixel_pitch_y_mm
+        data_dict["detector_heigth_mm"] = projection.detector_heigth_mm
+        data_dict["detector_width_mm"] = projection.detector_width_mm
 
         # xray
-        data_dict['exposure_time_ms'] = projection.exposure_time_ms
-        data_dict['voltage_kv'] = projection.voltage_kv
-        data_dict['current_ua'] = projection.current_ua
-        data_dict['frame_id'] = projection.frame_id
+        data_dict["exposure_time_ms"] = projection.exposure_time_ms
+        data_dict["voltage_kv"] = projection.voltage_kv
+        data_dict["current_ua"] = projection.current_ua
+        data_dict["frame_id"] = projection.frame_id
+
+        # extra / converted information
+        if optional:
+            data_dict["optional"] = dict(
+                projection_matrix_cera=projection.projection_matrix_cera_px.tolist()
+            )
 
         self.write_json(save_path_projection_geometry, data_dict)
 
         image = Image.fromarray(projection.image)
         image.save(save_path)
 
-    def write_region_of_intrest(self, save_path: Path, region_of_intrest: PyRegionOfIntrest):
+    def write_region_of_intrest(
+        self, save_path: Path, region_of_intrest: PyRegionOfIntrest
+    ):
         data_dict = dict()
 
         # geometry
-        data_dict['center_points_mm'] = region_of_intrest.center_points_mm.tolist()
-        data_dict['dimensions_mm'] = region_of_intrest.dimensions_mm.tolist()
-        data_dict['frame_id'] = region_of_intrest.frame_id
-        data_dict['resolution_mm'] = region_of_intrest.resolution_mm.tolist()
+        data_dict["center_points_mm"] = region_of_intrest.center_points_mm.tolist()
+        data_dict["dimensions_mm"] = region_of_intrest.dimensions_mm.tolist()
+        data_dict["frame_id"] = region_of_intrest.frame_id
+        data_dict["resolution_mm"] = region_of_intrest.resolution_mm.tolist()
 
         self.write_json(save_path, data_dict)
 
     def write_volume(self, save_path: Path, volume: PyVolume):
-        save_path_region_of_intrest = save_path.parent / f'{save_path.stem}{self.region_of_intrest_suffix}'
+        save_path_region_of_intrest = (
+            save_path.parent / f"{save_path.stem}{self.region_of_intrest_suffix}"
+        )
         self.write_region_of_intrest(save_path_region_of_intrest, volume.roi)
         writer = pyometiff.OMETIFFWriter(save_path, volume.array, dict())
-        writer.write()
\ No newline at end of file
+        writer.write()
diff --git a/rq_controller/common/io/thd/__init__.py b/rq_controller/common/io/thd/__init__.py
new file mode 100644
index 0000000..84f08e8
--- /dev/null
+++ b/rq_controller/common/io/thd/__init__.py
@@ -0,0 +1,5 @@
+from .thd_load import RawLoader
+from .thd_write import RawWriter
+
+
+__all__ = ["RawLoader", "RawWriter"]
diff --git a/rq_controller/common/io/thd/thd_load.py b/rq_controller/common/io/thd/thd_load.py
new file mode 100644
index 0000000..257d6cb
--- /dev/null
+++ b/rq_controller/common/io/thd/thd_load.py
@@ -0,0 +1,125 @@
+import numpy as np
+import json
+from pathlib import Path
+
+try:
+    from PythonTools.raw2py import raw2py
+    from PythonTools.ezrt_header import EzrtHeader
+    from PythonTools.rek2py import rek2py
+except ModuleNotFoundError:
+    raise ModuleNotFoundError("Install PythonTools from Fraunhofer EZRT.")
+
+from ..loader import (
+    BaseDataLoader,
+    PyProjection,
+    PyProjectionGeometry,
+    PyRegionOfIntrest,
+    PyVolume,
+)
+from scipy.spatial.transform import Rotation
+
+from ..rq_json.json_load import RqJsonLoader
+
+
+class RawLoader(BaseDataLoader):
+    def __init__(self):
+        super().__init__(".raw", ".raw", ".roi-json", ".rek")
+
+    def load_projection_geometry(self, load_path: Path) -> PyProjectionGeometry:
+        header = EzrtHeader.fromfile(load_path)
+
+        focal_spot_mm = np.array(header.agv_source_position) * 1000.0
+        detector_postion_mm = np.array(header.agv_detector_center_position) * 1000.0
+
+        line = np.array(header.agv_detector_line_direction)
+        column = np.array(header.agv_detector_col_direction)
+        normal = np.cross(line, column)
+
+        matrix = np.eye(3)
+        matrix[:, 0] = line
+        matrix[:, 1] = column
+        matrix[:, 2] = normal
+
+        rotation = Rotation.from_matrix(matrix)
+        detector_orientation_quad = rotation.as_quat()
+
+        frame_id = header.number_of_images
+
+        projection_geometry = PyProjectionGeometry(
+            focal_spot_mm,
+            detector_postion_mm,
+            detector_orientation_quad,
+            frame_id,
+            focal_spot_orientation_quad=np.array([0.0, 0.0, 0.0, 1.0]),
+        )
+
+        return projection_geometry
+
+    def load_projection(
+        self, load_path: Path, switch_order: bool = True
+    ) -> PyProjection:
+        header, image = raw2py(load_path, switch_order=switch_order)
+
+        focal_spot_mm = np.array(header.agv_source_position) * 1000.0
+        detector_postion_mm = np.array(header.agv_detector_center_position) * 1000.0
+
+        line = np.array(header.agv_detector_line_direction)
+        column = np.array(header.agv_detector_col_direction)
+        normal = np.cross(line, column)
+
+        matrix = np.eye(3)
+        matrix[:, 0] = line
+        matrix[:, 1] = column
+        matrix[:, 2] = normal
+
+        rotation = Rotation.from_matrix(matrix)
+        detector_orientation_quad = rotation.as_quat()
+
+        frame_id = header.number_of_images
+
+        detector_heigth_mm = header.pixel_width_in_um / 1000.0 * image.shape[1]
+        detector_width_mm = header.pixel_width_in_um / 1000.0 * image.shape[0]
+
+        voltage_kv = header.voltage_in_kv
+        current_ua = header.current_in_ua
+        exposure_time_ms = header.exposure_time_in_ms
+
+        projection = PyProjection(
+            focal_spot_mm,
+            detector_postion_mm,
+            detector_orientation_quad,
+            image,
+            detector_heigth_mm,
+            detector_width_mm,
+            frame_id,
+            np.array([0.0, 0.0, 0.0, 1.0]),
+            voltage_kv,
+            current_ua,
+            exposure_time_ms,
+        )
+
+        return projection
+
+    def load_json(self, load_path: Path) -> dict:
+        with open(str(load_path), "r") as f:
+            data_dict = json.load(f)
+        return data_dict
+
+    def load_region_of_intrest(self, load_path: Path) -> PyRegionOfIntrest:
+        RqJsonLoader.load_region_of_intrest(self, load_path)
+
+    def load_volume(self, load_path: Path) -> PyVolume:
+        header, volume = rek2py(load_path)
+        load_path_roi = (
+            load_path.parent / f"{load_path.stem}{self.region_of_intrest_suffix}"
+        )
+        roi = self.load_region_of_intrest(load_path_roi)
+
+        if volume.dtype == np.uint16:
+            data_type = 0
+        elif volume.dtype == np.uint8:
+            data_type = 1
+        else:
+            raise ValueError("data type not implemented.")
+
+        return PyVolume(volume, roi, data_type)
diff --git a/rq_controller/common/io/thd/thd_write.py b/rq_controller/common/io/thd/thd_write.py
new file mode 100644
index 0000000..ee0f3bf
--- /dev/null
+++ b/rq_controller/common/io/thd/thd_write.py
@@ -0,0 +1,86 @@
+from pathlib import Path
+
+from ..writer import (
+    BaseDataWriter,
+    PyProjection,
+    PyProjectionGeometry,
+    PyRegionOfIntrest,
+    PyVolume,
+)
+
+try:
+    from PythonTools.py2raw import py2raw
+    from PythonTools.ezrt_header import EzrtHeader
+    from PythonTools.py2rek import py2rek
+except ModuleNotFoundError:
+    raise ModuleNotFoundError("Install PythonTools from Fraunhofer EZRT.")
+
+
+from ..rq_json.json_write import RqJsonWriter
+
+
+class RawWriter(BaseDataWriter):
+    def __init__(self):
+        super().__init__(".raw", ".raw", ".roi-json", ".rek")
+
+    def write_projection_geometry(
+        self, save_path: Path, projection_geometry: PyProjectionGeometry
+    ):
+        header = EzrtHeader()
+        header.agv_source_position = (
+            projection_geometry.focal_spot_mm / 1000.0
+        ).tolist()
+        header.agv_detector_center_position = (
+            projection_geometry.detector_postion_mm / 1000.0
+        ).tolist()
+
+        line = projection_geometry.detector_rotation_matrix[:, 0]
+        column = projection_geometry.detector_rotation_matrix[:, 0]
+
+        header.agv_detector_line_direction = line
+        header.agv_detector_col_direction = column
+
+        header.number_of_images = projection_geometry.frame_id
+        header.tofile(save_path)
+
+    def write_projection(
+        self, save_path: Path, projection: PyProjection, swith_order: bool = False
+    ):
+        projection_geometry = projection.get_projection_geometry()
+
+        header = EzrtHeader()
+
+        header.agv_source_position = (
+            projection_geometry.focal_spot_mm / 1000.0
+        ).tolist()
+        header.agv_detector_center_position = (
+            projection_geometry.detector_postion_mm / 1000.0
+        ).tolist()
+
+        line = projection_geometry.detector_rotation_matrix[:, 0]
+        column = projection_geometry.detector_rotation_matrix[:, 0]
+
+        header.agv_detector_line_direction = line
+        header.agv_detector_col_direction = column
+
+        header.number_of_images = projection_geometry.frame_id
+        header.pixel_width_in_um = projection.pixel_pitch_x_mm * 1000.0
+        header.voltage_in_kv = projection.voltage_kv
+        header.current_in_ua = projection.current_ua
+        header.exposure_time_in_ms = projection.exposure_time_ms
+
+        py2raw(projection.image, save_path, header, swith_order)
+
+    def write_region_of_intrest(
+        self, save_path: Path, region_of_intrest: PyRegionOfIntrest
+    ):
+        RqJsonWriter.write_region_of_intrest(self, save_path, region_of_intrest)
+
+    def write_volume(self, save_path: Path, volume: PyVolume):
+        header = EzrtHeader()
+        header.num_voxel_x = volume.shape[0]
+        header.num_voxel_y = volume.shape[1]
+        header.num_voxel_z = volume.shape[2]
+        header.pixel_width_in_um = volume.roi.resolution_mm[0]
+
+        py2rek(volume.array, save_path, header, True)
diff --git a/rq_controller/common/io/writer.py b/rq_controller/common/io/writer.py
index f9475c3..91ae744 100644
--- a/rq_controller/common/io/writer.py
+++ b/rq_controller/common/io/writer.py
@@ -1,70 +1,101 @@
-import numpy as np
-import json
 from pathlib import Path
 from ...common import PyProjectionGeometry, PyProjection, PyRegionOfIntrest, PyVolume
 
 
-class BaseDataWriter():
-    projection_name: str = 'projection'
-    projection_geometry_name: str ='geometry'
-    region_of_intrest_name: str = 'roi'
-    volume_name: str = 'volume'
+class BaseDataWriter:
+    projection_name: str = "projection"
+    projection_geometry_name: str = "geometry"
+    region_of_intrest_name: str = "roi"
+    volume_name: str = "volume"
 
-    def __init__(self, 
-                 porjection_geometry_suffix: str, 
-                 projection_suffix: str,
-                 region_of_intrest_suffix: str,
-                 volume_suffix: str):
-        
+    def __init__(
+        self,
+        porjection_geometry_suffix: str,
+        projection_suffix: str,
+        region_of_intrest_suffix: str,
+        volume_suffix: str,
+    ):
         self.porjection_geometry_suffix = porjection_geometry_suffix
         self.projection_suffix = projection_suffix
         self.region_of_intrest_suffix = region_of_intrest_suffix
         self.volume_suffix = volume_suffix
 
-    def write_projection_geometry(self, save_path: Path, projection_geometry: PyProjectionGeometry):
+    def write_projection_geometry(
+        self, save_path: Path, projection_geometry: PyProjectionGeometry
+    ):
         raise NotImplementedError()
-    
+
     def write_projection(self, save_path: Path, projection: PyProjection):
         raise NotImplementedError()
-    
-    def write_region_of_intrest(self, save_path: Path, region_of_intrest: PyRegionOfIntrest):
+
+    def write_region_of_intrest(
+        self, save_path: Path, region_of_intrest: PyRegionOfIntrest
+    ):
         raise NotImplementedError
-    
+
     def write_volume(self, save_path: Path, volume: PyVolume):
         raise NotImplementedError()
-    
+
     def get_next_projection_save_path(self, save_folder: Path) -> Path:
-        return self.get_projection_save_path_i(save_folder, self.number_of_projections(save_folder) + 1)
+        return self.get_projection_save_path_i(
+            save_folder, self.number_of_projections(save_folder) + 1
+        )
 
     def get_projection_save_path_i(self, save_folder: Path, i) -> Path:
-        return save_folder / f'{self.projection_name}_{i:05}{self.projection_suffix}'
-    
+        return save_folder / f"{self.projection_name}_{i:05}{self.projection_suffix}"
+
     def get_next_projection_geometry_save_path(self, save_folder: Path) -> Path:
-        return self.get_projection_geometry_save_path_i(save_folder, self.number_of_projection_geometries(save_folder) + 1)
+        return self.get_projection_geometry_save_path_i(
+            save_folder, self.number_of_projection_geometries(save_folder) + 1
+        )
 
     def get_projection_geometry_save_path_i(self, save_folder: Path, i) -> Path:
-        return save_folder / f'{self.projection_geometry_name}_{i:05}{self.porjection_geometry_suffix}'
-    
+        return (
+            save_folder
+            / f"{self.projection_geometry_name}_{i:05}{self.porjection_geometry_suffix}"
+        )
+
     def get_next_region_of_intrest_save_path(self, save_folder: Path) -> Path:
-        return self.get_region_of_intrest_save_path_i(save_folder, self.number_of_region_of_intrests(save_folder) + 1)
+        return self.get_region_of_intrest_save_path_i(
+            save_folder, self.number_of_region_of_intrests(save_folder) + 1
+        )
 
     def get_region_of_intrest_save_path_i(self, save_folder: Path, i) -> Path:
-        return save_folder / f'{self.region_of_intrest_name}_{i:05}{self.region_of_intrest_suffix}'
-    
+        return (
+            save_folder
+            / f"{self.region_of_intrest_name}_{i:05}{self.region_of_intrest_suffix}"
+        )
+
     def get_next_volume_save_path(self, save_folder: Path) -> Path:
-        return self.get_volume_save_path_i(save_folder, self.number_of_volumes(save_folder) + 1)
+        return self.get_volume_save_path_i(
+            save_folder, self.number_of_volumes(save_folder) + 1
+        )
 
     def get_volume_save_path_i(self, save_folder: Path, i) -> Path:
-        return save_folder / f'{self.volume_name}_{i:05}{self.volume_suffix}'
-    
+        return save_folder / f"{self.volume_name}_{i:05}{self.volume_suffix}"
+
     def number_of_projection_geometries(self, folder: Path) -> int:
-        return len(list(folder.glob(f'{self.projection_geometry_name}*{self.porjection_geometry_suffix}')))
-    
+        return len(
+            list(
+                folder.glob(
+                    f"{self.projection_geometry_name}*{self.porjection_geometry_suffix}"
+                )
+            )
+        )
+
     def number_of_projections(self, folder: Path) -> int:
-        return len(list(folder.glob(f'{self.projection_name}*{self.projection_suffix}')))
-    
+        return len(
+            list(folder.glob(f"{self.projection_name}*{self.projection_suffix}"))
+        )
+
     def number_of_region_of_intrests(self, folder: Path) -> int:
-        return len(list(folder.glob(f'{self.region_of_intrest_name}*{self.region_of_intrest_suffix}')))
-    
+        return len(
+            list(
+                folder.glob(
+                    f"{self.region_of_intrest_name}*{self.region_of_intrest_suffix}"
+                )
+            )
+        )
+
     def number_of_volumes(self, folder: Path) -> int:
-        return len(list(folder.glob(f'{self.volume_name}*{self.volume_suffix}')))
+        return len(list(folder.glob(f"{self.volume_name}*{self.volume_suffix}")))
diff --git a/rq_controller/common/projection.py b/rq_controller/common/projection.py
index d69ab60..2f81509 100644
--- a/rq_controller/common/projection.py
+++ b/rq_controller/common/projection.py
@@ -24,10 +24,20 @@ class PyProjection(PyProjectionGeometry):
         exposure_time_ms (float): The exposure time in milliseconds.
     """
 
-    def __init__(self, focal_spot_mm: ndarray, detector_postion_mm: ndarray, detector_orientation_quad: ndarray, image: np.ndarray,
-                 detector_heigth_mm: float, detector_width_mm: float, frame_id: str = 'object', 
-                 focal_spot_orientation_quad: np.ndarray = np.array([0., 0., 0, 1.]),
-                 voltage_kv: float = 100., current_ua: float = 100., exposure_time_ms: float = 1000.) -> None:
+    def __init__(
+        self,
+        focal_spot_mm: ndarray,
+        detector_postion_mm: ndarray,
+        detector_orientation_quad: ndarray,
+        image: np.ndarray,
+        detector_heigth_mm: float,
+        detector_width_mm: float,
+        frame_id: str = "object",
+        focal_spot_orientation_quad: np.ndarray = np.array([0.0, 0.0, 0, 1.0]),
+        voltage_kv: float = 100.0,
+        current_ua: float = 100.0,
+        exposure_time_ms: float = 1000.0,
+    ) -> None:
         """
         Initializes a PyProjection instance.
 
@@ -44,8 +54,14 @@ class PyProjection(PyProjectionGeometry):
             current_ua (float): The current in microamperes. Default is 100.
             exposure_time_ms (float): The exposure time in milliseconds. Default is 1000.
         """
-        
-        super().__init__(focal_spot_mm, detector_postion_mm, detector_orientation_quad, frame_id, focal_spot_orientation_quad)
+
+        super().__init__(
+            focal_spot_mm,
+            detector_postion_mm,
+            detector_orientation_quad,
+            frame_id,
+            focal_spot_orientation_quad,
+        )
         self.image = image.astype(np.uint16)
         self.detector_heigth_mm = detector_heigth_mm
         self.detector_width_mm = detector_width_mm
@@ -62,19 +78,31 @@ class PyProjection(PyProjectionGeometry):
             PyProjection: A dummy instance with default values.
         """
 
-        return cls(np.array([0., 100., 0]), 
-                   np.array([0., -100., 0]),
-                   np.array([1., 0., 0, 1.]),
-                   np.random.randint(0, 65535, size=(10, 10), dtype=np.uint16),
-                   10., 10.)
-    
+        return cls(
+            np.array([0.0, 100.0, 0]),
+            np.array([0.0, -100.0, 0]),
+            np.array([1.0, 0.0, 0, 1.0]),
+            np.random.randint(0, 65535, size=(10, 10), dtype=np.uint16),
+            10.0,
+            10.0,
+        )
+
     @classmethod
-    def from_look_at(cls, focal_spot_mm: ndarray, detector_postion_mm: ndarray, image_shape: tuple[int],
-                          detector_heigth_mm: float, detector_width_mm: float, frame_id: str = 'object', 
-                          voltage_kv: float = 100., current_ua: float = 100., exposure_time_ms: float = 1000.,
-                          up_vector: np.ndarray = np.array([0., 0., 1.])) -> 'PyProjection':
+    def from_look_at(
+        cls,
+        focal_spot_mm: ndarray,
+        detector_postion_mm: ndarray,
+        image_shape: tuple[int],
+        detector_heigth_mm: float,
+        detector_width_mm: float,
+        frame_id: str = "object",
+        voltage_kv: float = 100.0,
+        current_ua: float = 100.0,
+        exposure_time_ms: float = 1000.0,
+        up_vector: np.ndarray = np.array([0.0, 0.0, 1.0]),
+    ) -> "PyProjection":
         image = np.zeros(image_shape, np.uint16)
-        
+
         vector = focal_spot_mm - detector_postion_mm
         vector = vector / np.linalg.norm(vector)
         up_vector = up_vector / np.linalg.norm(up_vector)
@@ -82,47 +110,78 @@ class PyProjection(PyProjectionGeometry):
         detector_orientation = np.eye(3)
         detector_orientation[:, 2] = vector
         detector_orientation[:, 1] = np.cross(up_vector, vector)
-        detector_orientation[:, 0] = np.cross(detector_orientation[:, 1], detector_orientation[:, 2])
+        detector_orientation[:, 0] = np.cross(
+            detector_orientation[:, 1], detector_orientation[:, 2]
+        )
 
         source_orientation = np.eye(3)
         source_orientation[:, 2] = -vector
         source_orientation[:, 1] = np.cross(up_vector, -vector)
-        source_orientation[:, 0] = np.cross(source_orientation[:, 1], source_orientation[:, 2])
+        source_orientation[:, 0] = np.cross(
+            source_orientation[:, 1], source_orientation[:, 2]
+        )
 
         return cls(
-            focal_spot_mm, detector_postion_mm,
+            focal_spot_mm,
+            detector_postion_mm,
             Rotation.from_matrix(detector_orientation).as_quat(),
-            image, detector_heigth_mm, detector_width_mm,
-            frame_id, Rotation.from_matrix(source_orientation).as_quat(),
-            voltage_kv, current_ua, exposure_time_ms)
-    
-    def look_at(self, focal_spot_mm: ndarray, detector_postion_mm: ndarray, up_vector: np.ndarray = np.array([0., 0., 1.])) -> PyProjection:
+            image,
+            detector_heigth_mm,
+            detector_width_mm,
+            frame_id,
+            Rotation.from_matrix(source_orientation).as_quat(),
+            voltage_kv,
+            current_ua,
+            exposure_time_ms,
+        )
+
+    def look_at(
+        self,
+        focal_spot_mm: ndarray,
+        detector_postion_mm: ndarray,
+        up_vector: np.ndarray = np.array([0.0, 0.0, 1.0]),
+    ) -> PyProjection:
         vector = focal_spot_mm - detector_postion_mm
         vector = vector / np.linalg.norm(vector)
         up_vector = up_vector / np.linalg.norm(up_vector)
 
-        z_rot = Rotation.from_euler('Z', -90, True)
+        z_rot = Rotation.from_euler("Z", -90, True)
 
         detector_orientation = np.eye(3)
         detector_orientation[:, 2] = vector
         detector_orientation[:, 1] = np.cross(up_vector, vector)
-        detector_orientation[:, 0] = np.cross(detector_orientation[:, 1], detector_orientation[:, 2])
+        detector_orientation[:, 0] = np.cross(
+            detector_orientation[:, 1], detector_orientation[:, 2]
+        )
 
-        detector_orientation_quad = (Rotation.from_matrix(detector_orientation) * z_rot).as_quat()
+        detector_orientation_quad = (
+            Rotation.from_matrix(detector_orientation) * z_rot
+        ).as_quat()
 
         source_orientation = np.eye(3)
         source_orientation[:, 2] = -vector
         source_orientation[:, 1] = np.cross(up_vector, -vector)
-        source_orientation[:, 0] = np.cross(source_orientation[:, 1], source_orientation[:, 2])
-        source_orientation_quad = (Rotation.from_matrix(source_orientation) * z_rot).as_quat()
-        
+        source_orientation[:, 0] = np.cross(
+            source_orientation[:, 1], source_orientation[:, 2]
+        )
+        source_orientation_quad = (
+            Rotation.from_matrix(source_orientation) * z_rot
+        ).as_quat()
+
         return PyProjection(
-            focal_spot_mm, detector_postion_mm, detector_orientation_quad, self.image,
-            self.detector_heigth_mm, self.detector_width_mm,
-            self.frame_id, source_orientation_quad, self.voltage_kv, 
-            self.current_ua, self.exposure_time_ms)
+            focal_spot_mm,
+            detector_postion_mm,
+            detector_orientation_quad,
+            self.image,
+            self.detector_heigth_mm,
+            self.detector_width_mm,
+            self.frame_id,
+            source_orientation_quad,
+            self.voltage_kv,
+            self.current_ua,
+            self.exposure_time_ms,
+        )
 
-    
     def __str__(self) -> str:
         """
         Returns a string representation of the PyProjection instance.
@@ -130,14 +189,18 @@ class PyProjection(PyProjectionGeometry):
         Returns:
             str: The string representation of the projection.
         """
-        print_str = f'---\n'
-        print_str += f'PyProjection\n'
-        print_str += f'--- Projection Geometry:\n'
-        print_str += f'Detector Posiion [mm]:         \t {self.detector_postion_mm.tolist()} \n'
-        print_str += f'Focal Spot [mm]:               \t {self.focal_spot_mm.tolist()} \n'
-        print_str += f'Detector Orientation [quad]:   \t {self.detector_orientation_quad.tolist()} \n'
-        print_str += f'Focal Spot Orientation [quad]: \t {self.focal_spot_orientation_quad.tolist()} \n'
-        print_str += f'---\n\n'
+        print_str = "---\n"
+        print_str += "PyProjection\n"
+        print_str += "--- Projection Geometry:\n"
+        print_str += (
+            f"Detector Posiion [mm]:         \t {self.detector_postion_mm.tolist()} \n"
+        )
+        print_str += (
+            f"Focal Spot [mm]:               \t {self.focal_spot_mm.tolist()} \n"
+        )
+        print_str += f"Detector Orientation [quad]:   \t {self.detector_orientation_quad.tolist()} \n"
+        print_str += f"Focal Spot Orientation [quad]: \t {self.focal_spot_orientation_quad.tolist()} \n"
+        print_str += "---\n\n"
         return print_str
 
     @classmethod
@@ -151,25 +214,40 @@ class PyProjection(PyProjectionGeometry):
         Returns:
             PyProjection: An instance initialized from the ROS message.
         """
-        focal_spot_mm = np.array([msg.projection_geometry.focal_spot_postion_mm.x,
-                                  msg.projection_geometry.focal_spot_postion_mm.y,
-                                  msg.projection_geometry.focal_spot_postion_mm.z,])
-
-        detector_center_mm = np.array([msg.projection_geometry.detector_postion_mm.x,
-                                       msg.projection_geometry.detector_postion_mm.y,
-                                       msg.projection_geometry.detector_postion_mm.z,])
-        
-        detector_orientation_quad = np.array([msg.projection_geometry.detector_orientation_quad.x,
-                                              msg.projection_geometry.detector_orientation_quad.y,
-                                              msg.projection_geometry.detector_orientation_quad.z,
-                                              msg.projection_geometry.detector_orientation_quad.w])
-        
-        focal_spot_orientation = np.array([msg.projection_geometry.focal_spot_orientation_quad.x,
-                                           msg.projection_geometry.focal_spot_orientation_quad.y,
-                                           msg.projection_geometry.focal_spot_orientation_quad.z,
-                                           msg.projection_geometry.focal_spot_orientation_quad.w])
-        
-        
+        focal_spot_mm = np.array(
+            [
+                msg.projection_geometry.focal_spot_postion_mm.x,
+                msg.projection_geometry.focal_spot_postion_mm.y,
+                msg.projection_geometry.focal_spot_postion_mm.z,
+            ]
+        )
+
+        detector_center_mm = np.array(
+            [
+                msg.projection_geometry.detector_postion_mm.x,
+                msg.projection_geometry.detector_postion_mm.y,
+                msg.projection_geometry.detector_postion_mm.z,
+            ]
+        )
+
+        detector_orientation_quad = np.array(
+            [
+                msg.projection_geometry.detector_orientation_quad.x,
+                msg.projection_geometry.detector_orientation_quad.y,
+                msg.projection_geometry.detector_orientation_quad.z,
+                msg.projection_geometry.detector_orientation_quad.w,
+            ]
+        )
+
+        focal_spot_orientation = np.array(
+            [
+                msg.projection_geometry.focal_spot_orientation_quad.x,
+                msg.projection_geometry.focal_spot_orientation_quad.y,
+                msg.projection_geometry.focal_spot_orientation_quad.z,
+                msg.projection_geometry.focal_spot_orientation_quad.w,
+            ]
+        )
+
         detector_heigth_mm = msg.detector_heigth_mm
         detector_width_mm = msg.detector_width_mm
         frame_id = msg.projection_geometry.header.frame_id
@@ -180,10 +258,20 @@ class PyProjection(PyProjectionGeometry):
         current_ua = msg.current_ua
         exposure_time_ms = msg.exposure_time_ms
 
-        return cls(focal_spot_mm, detector_center_mm, detector_orientation_quad, image, 
-                   detector_heigth_mm, detector_width_mm, frame_id, focal_spot_orientation,
-                   voltage_kv, current_ua, exposure_time_ms)
-    
+        return cls(
+            focal_spot_mm,
+            detector_center_mm,
+            detector_orientation_quad,
+            image,
+            detector_heigth_mm,
+            detector_width_mm,
+            frame_id,
+            focal_spot_orientation,
+            voltage_kv,
+            current_ua,
+            exposure_time_ms,
+        )
+
     def as_message(self) -> Projection:
         """
         Converts the PyProjection instance to a ROS message.
@@ -197,7 +285,9 @@ class PyProjection(PyProjectionGeometry):
         self.focal_spot_mm = self.focal_spot_mm.reshape((3,))
         self.detector_postion_mm = self.detector_postion_mm.reshape((3,))
         self.detector_orientation_quad = self.detector_orientation_quad.reshape((4,))
-        self.focal_spot_orientation_quad = self.focal_spot_orientation_quad.reshape((4,))
+        self.focal_spot_orientation_quad = self.focal_spot_orientation_quad.reshape(
+            (4,)
+        )
 
         projection_geometry.focal_spot_postion_mm.x = float(self.focal_spot_mm[0])
         projection_geometry.focal_spot_postion_mm.y = float(self.focal_spot_mm[1])
@@ -207,18 +297,34 @@ class PyProjection(PyProjectionGeometry):
         projection_geometry.detector_postion_mm.y = float(self.detector_postion_mm[1])
         projection_geometry.detector_postion_mm.z = float(self.detector_postion_mm[2])
 
-        projection_geometry.detector_orientation_quad.x = float(self.detector_orientation_quad[0])
-        projection_geometry.detector_orientation_quad.y = float(self.detector_orientation_quad[1])
-        projection_geometry.detector_orientation_quad.z = float(self.detector_orientation_quad[2])
-        projection_geometry.detector_orientation_quad.w = float(self.detector_orientation_quad[3])
-
-        projection_geometry.focal_spot_orientation_quad.x = float(self.focal_spot_orientation_quad[0])
-        projection_geometry.focal_spot_orientation_quad.y = float(self.focal_spot_orientation_quad[1])
-        projection_geometry.focal_spot_orientation_quad.z = float(self.focal_spot_orientation_quad[2])
-        projection_geometry.focal_spot_orientation_quad.w = float(self.focal_spot_orientation_quad[3])
+        projection_geometry.detector_orientation_quad.x = float(
+            self.detector_orientation_quad[0]
+        )
+        projection_geometry.detector_orientation_quad.y = float(
+            self.detector_orientation_quad[1]
+        )
+        projection_geometry.detector_orientation_quad.z = float(
+            self.detector_orientation_quad[2]
+        )
+        projection_geometry.detector_orientation_quad.w = float(
+            self.detector_orientation_quad[3]
+        )
+
+        projection_geometry.focal_spot_orientation_quad.x = float(
+            self.focal_spot_orientation_quad[0]
+        )
+        projection_geometry.focal_spot_orientation_quad.y = float(
+            self.focal_spot_orientation_quad[1]
+        )
+        projection_geometry.focal_spot_orientation_quad.z = float(
+            self.focal_spot_orientation_quad[2]
+        )
+        projection_geometry.focal_spot_orientation_quad.w = float(
+            self.focal_spot_orientation_quad[3]
+        )
 
         message.projection_geometry = projection_geometry
-        message.image = ros2_numpy.msgify(Image, self.image, 'mono16')
+        message.image = ros2_numpy.msgify(Image, self.image, "mono16")
 
         message.detector_heigth_mm = self.detector_heigth_mm
         message.detector_width_mm = self.detector_width_mm
@@ -230,28 +336,46 @@ class PyProjection(PyProjectionGeometry):
         message.exposure_time_ms = self.exposure_time_ms
 
         return message
-    
+
     def get_projection_geometry(self) -> PyProjectionGeometry:
-        return PyProjectionGeometry(self.focal_spot_mm,
-                                    self.detector_postion_mm,
-                                    self.detector_orientation_quad,
-                                    self.frame_id,
-                                    self.focal_spot_orientation_quad)
-    
+        return PyProjectionGeometry(
+            self.focal_spot_mm,
+            self.detector_postion_mm,
+            self.detector_orientation_quad,
+            self.frame_id,
+            self.focal_spot_orientation_quad,
+        )
+
     @property
     def detector_heigth_px(self) -> int:
         return self.image.shape[0]
-    
+
     @property
     def detector_width_px(self) -> int:
         return self.image.shape[1]
-    
+
     @property
     def pixel_pitch_x_mm(self) -> float:
         return self.detector_width_mm / self.detector_width_px
-    
+
     @property
     def pixel_pitch_y_mm(self) -> float:
         return self.detector_heigth_mm / self.detector_heigth_px
-    
-    
+
+    @property
+    def to_cera_transformation_matrix(self) -> np.ndarray:
+        transformation_matrix = np.eye(3)
+        # [mm] -> [px]
+        transformation_matrix[2, 2] = 1.0 / self.pixel_pitch_x_mm
+        transformation_matrix[0, 2] = -self.detector_width_px / 2.0
+        transformation_matrix[1, 2] = -self.detector_heigth_px / 2.0
+
+        return transformation_matrix
+
+    @property
+    def projection_matrix_cera_px(self):
+        projection_geometry = self.get_projection_geometry()
+        return (
+            np.linalg.inv(self.to_cera_transformation_matrix)
+            @ projection_geometry.projection_matrix
+        )
diff --git a/rq_controller/common/projection_geometry.py b/rq_controller/common/projection_geometry.py
index 61bc500..2c94ff6 100644
--- a/rq_controller/common/projection_geometry.py
+++ b/rq_controller/common/projection_geometry.py
@@ -2,9 +2,11 @@ from __future__ import annotations
 
 import numpy as np
 
-from rq_interfaces.msg import ProjectionGeometry, Projection
+from rq_interfaces.msg import ProjectionGeometry
+from scipy.spatial.transform import Rotation
 
-class PyProjectionGeometry():
+
+class PyProjectionGeometry:
     """
     Represents the geometry of a projection including the focal spot and detector position and orientation.
 
@@ -16,10 +18,14 @@ class PyProjectionGeometry():
         frame_id (str): Frame ID for the projection geometry.
     """
 
-    def __init__(self, focal_spot_mm: np.ndarray, detector_postion_mm: np.ndarray, 
-                 detector_orientation_quad: np.ndarray,
-                 frame_id: str = 'object', focal_spot_orientation_quad: np.ndarray = np.array([0., 0., 0, 1.])
-                 ) -> None:
+    def __init__(
+        self,
+        focal_spot_mm: np.ndarray,
+        detector_postion_mm: np.ndarray,
+        detector_orientation_quad: np.ndarray,
+        frame_id: str = "object",
+        focal_spot_orientation_quad: np.ndarray = np.array([0.0, 0.0, 0, 1.0]),
+    ) -> None:
         """
         Initializes a PyProjectionGeometry instance.
 
@@ -45,9 +51,11 @@ class PyProjectionGeometry():
         Returns:
             PyProjectionGeometry: A dummy instance with default values.
         """
-        return cls(np.array([1., 0., 0]), 
-                   np.array([-1., 0., 0]),
-                   np.array([0., 0., 0, 1.]))
+        return cls(
+            np.array([1.0, 0.0, 0]),
+            np.array([-1.0, 0.0, 0]),
+            np.array([0.0, 0.0, 0, 1.0]),
+        )
 
     @classmethod
     def from_message(cls, msg: ProjectionGeometry):
@@ -60,28 +68,50 @@ class PyProjectionGeometry():
         Returns:
             PyProjectionGeometry: An instance initialized from the ROS message.
         """
-        focal_spot_mm = np.array([msg.focal_spot_postion_mm.x,
-                                  msg.focal_spot_postion_mm.y,
-                                  msg.focal_spot_postion_mm.z,])
-
-        detector_center_mm = np.array([msg.detector_postion_mm.x,
-                                       msg.detector_postion_mm.y,
-                                       msg.detector_postion_mm.z,])
-        
-        detector_orientation_quad = np.array([msg.detector_orientation_quad.x,
-                                              msg.detector_orientation_quad.y,
-                                              msg.detector_orientation_quad.z,
-                                              msg.detector_orientation_quad.w])
-        
-        focal_spot_orientation = np.array([msg.focal_spot_orientation_quad.x,
-                                           msg.focal_spot_orientation_quad.y,
-                                           msg.focal_spot_orientation_quad.z,
-                                           msg.focal_spot_orientation_quad.w])
-        
+        focal_spot_mm = np.array(
+            [
+                msg.focal_spot_postion_mm.x,
+                msg.focal_spot_postion_mm.y,
+                msg.focal_spot_postion_mm.z,
+            ]
+        )
+
+        detector_center_mm = np.array(
+            [
+                msg.detector_postion_mm.x,
+                msg.detector_postion_mm.y,
+                msg.detector_postion_mm.z,
+            ]
+        )
+
+        detector_orientation_quad = np.array(
+            [
+                msg.detector_orientation_quad.x,
+                msg.detector_orientation_quad.y,
+                msg.detector_orientation_quad.z,
+                msg.detector_orientation_quad.w,
+            ]
+        )
+
+        focal_spot_orientation = np.array(
+            [
+                msg.focal_spot_orientation_quad.x,
+                msg.focal_spot_orientation_quad.y,
+                msg.focal_spot_orientation_quad.z,
+                msg.focal_spot_orientation_quad.w,
+            ]
+        )
+
         frame_id = msg.header.frame_id
 
-        return cls(focal_spot_mm, detector_center_mm, detector_orientation_quad, frame_id, focal_spot_orientation)
-    
+        return cls(
+            focal_spot_mm,
+            detector_center_mm,
+            detector_orientation_quad,
+            frame_id,
+            focal_spot_orientation,
+        )
+
     def as_message(self) -> ProjectionGeometry:
         """
         Converts the PyProjectionGeometry instance to a ROS message.
@@ -111,4 +141,30 @@ class PyProjectionGeometry():
 
         message.header.frame_id = self.frame_id
 
-        return message
\ No newline at end of file
+        return message
+
+    @property
+    def detector_rotation_matrix(self) -> np.ndarray:
+        return Rotation.from_quat(self.detector_orientation_quad).as_matrix()
+
+    @property
+    def detector_horizontal_vector(self) -> np.ndarray:
+        return self.detector_rotation_matrix[:, 0]
+
+    @property
+    def detector_vertical_vector(self) -> np.ndarray:
+        return self.detector_rotation_matrix[:, 1]
+
+    @property
+    def projection_matrix(self) -> np.ndarray:
+        p3x3 = np.vstack(
+            [
+                self.detector_horizontal_vector,
+                self.detector_vertical_vector,
+                self.detector_postion_mm - self.focal_spot_mm,
+            ]
+        ).T
+        p3x3_inv = np.linalg.inv(p3x3)
+        p4 = (p3x3_inv @ (-self.focal_spot_mm)).reshape((3, 1))
+        matrix = np.concatenate([p3x3_inv, p4], 1).reshape((3, 4))
+        return matrix
diff --git a/rq_controller/common/region_of_intrest.py b/rq_controller/common/region_of_intrest.py
index 4a71fda..e7cbe2f 100644
--- a/rq_controller/common/region_of_intrest.py
+++ b/rq_controller/common/region_of_intrest.py
@@ -6,7 +6,7 @@ from rq_interfaces.msg import RegionOfIntrest
 from visualization_msgs.msg import Marker
 
 
-class PyRegionOfIntrest():
+class PyRegionOfIntrest:
     """
     Represents a region of interest (ROI) with center points, dimensions, and resolution.
 
@@ -17,8 +17,13 @@ class PyRegionOfIntrest():
         resolution_mm (np.ndarray): Resolution of the ROIs in millimeters.
     """
 
-    def __init__(self, center_points_mm: np.ndarray, dimensions_mm: np.ndarray, frame_id: str = 'object', 
-                 resolution_mm: np.ndarray = np.array([0.1, 0.1, 0.1])):
+    def __init__(
+        self,
+        center_points_mm: np.ndarray,
+        dimensions_mm: np.ndarray,
+        frame_id: str = "object",
+        resolution_mm: np.ndarray = np.array([0.1, 0.1, 0.1]),
+    ):
         """
         Initializes a PyRegionOfIntrest instance.
 
@@ -43,8 +48,7 @@ class PyRegionOfIntrest():
             PyRegionOfIntrest: A dummy instance with random values.
         """
 
-        return cls((np.random.random((3, )) - 0.5) * 20., 
-                   np.random.random((3, )) * 10.)
+        return cls((np.random.random((3,)) - 0.5) * 20.0, np.random.random((3,)) * 10.0)
 
     @classmethod
     def from_message(cls, msg: RegionOfIntrest):
@@ -64,22 +68,20 @@ class PyRegionOfIntrest():
         for roi in msg.region_of_intrest_stack.markers:
             roi: Marker
             center_points_mm.append(
-                np.array([roi.pose.position.x,
-                          roi.pose.position.y,
-                          roi.pose.position.z]))
-            
-            dimensions_mm.append(
-                np.array([roi.scale.x,
-                          roi.scale.y,
-                          roi.scale.z]))
-            
+                np.array(
+                    [roi.pose.position.x, roi.pose.position.y, roi.pose.position.z]
+                )
+            )
+
+            dimensions_mm.append(np.array([roi.scale.x, roi.scale.y, roi.scale.z]))
+
             frame = roi.header.frame_id
-        resolution_mm = np.array([msg.resolution.x,
-                                  msg.resolution.y,
-                                  msg.resolution.z])
-        
-        return cls(np.array(center_points_mm), np.array(dimensions_mm), frame, resolution_mm)
-    
+        resolution_mm = np.array([msg.resolution.x, msg.resolution.y, msg.resolution.z])
+
+        return cls(
+            np.array(center_points_mm), np.array(dimensions_mm), frame, resolution_mm
+        )
+
     @property
     def number_of_rois(self) -> int:
         """
@@ -90,7 +92,7 @@ class PyRegionOfIntrest():
         """
 
         return self.center_points_mm.shape[0]
-    
+
     @property
     def shape(self) -> tuple:
         """
@@ -102,7 +104,7 @@ class PyRegionOfIntrest():
 
         shape = self.dimensions_mm[0] // self.resolution_mm[0]
         return (int(shape[0]), int(shape[1]), int(shape[2]))
-      
+
     def as_message(self) -> RegionOfIntrest:
         """
         Converts the PyRegionOfIntrest instance to a ROS message.
@@ -127,8 +129,8 @@ class PyRegionOfIntrest():
 
             roi.header.frame_id = self.frame_id
 
-            roi_list.append(roi)            
-        
+            roi_list.append(roi)
+
         message.region_of_intrest_stack.markers = roi_list
 
         message.resolution.x = float(self.resolution_mm[0][0])
@@ -136,7 +138,7 @@ class PyRegionOfIntrest():
         message.resolution.z = float(self.resolution_mm[0][2])
 
         return message
-    
+
     def get_grid(self, indice: int = 0) -> np.ndarray:
         """
         Generates a grid of points within the ROI.
@@ -148,21 +150,19 @@ class PyRegionOfIntrest():
             np.ndarray: A grid of points within the ROI.
         """
 
-        start = self.center_points_mm[indice] - (self.dimensions_mm[indice] / 2.)
-        end = self.center_points_mm[indice] + (self.dimensions_mm[indice] / 2.)
-        
+        start = self.center_points_mm[indice] - (self.dimensions_mm[indice] / 2.0)
+        end = self.center_points_mm[indice] + (self.dimensions_mm[indice] / 2.0)
+
         x_ = np.linspace(start[0], end[0], self.shape[0])
         y_ = np.linspace(start[1], end[1], self.shape[1])
         z_ = np.linspace(start[2], end[2], self.shape[2])
 
-        x, y, z = np.meshgrid(x_, y_, z_, indexing='ij')
+        x, y, z = np.meshgrid(x_, y_, z_, indexing="ij")
+
+        return np.concatenate(
+            (np.expand_dims(x, -1), np.expand_dims(y, -1), np.expand_dims(z, -1)), -1
+        )
 
-        return np.concatenate((
-            np.expand_dims(x, -1),
-            np.expand_dims(y, -1),
-            np.expand_dims(z, -1)),
-            -1)
-    
     @staticmethod
     def next_neighbor(grid_mm: np.ndarray, point_mm: np.ndarray) -> np.ndarray:
         """
@@ -175,13 +175,13 @@ class PyRegionOfIntrest():
         Returns:
             np.ndarray: The indices of the nearest neighbor in the grid.
         """
-        
+
         x = grid_mm[:, 0, 0, 0]
         y = grid_mm[0, :, 0, 1]
         z = grid_mm[0, 0, :, 2]
 
-        xx = int(np.argmin((x - point_mm[0])**2))
-        yy = int(np.argmin((y - point_mm[1])**2))
-        zz = int(np.argmin((z - point_mm[2])**2))
+        xx = int(np.argmin((x - point_mm[0]) ** 2))
+        yy = int(np.argmin((y - point_mm[1]) ** 2))
+        zz = int(np.argmin((z - point_mm[2]) ** 2))
 
-        return np.array([xx, yy, zz], dtype=np.int32)
\ No newline at end of file
+        return np.array([xx, yy, zz], dtype=np.int32)
diff --git a/rq_controller/common/volume.py b/rq_controller/common/volume.py
index ac1b645..b63aea9 100644
--- a/rq_controller/common/volume.py
+++ b/rq_controller/common/volume.py
@@ -19,7 +19,7 @@ class VOLUME_TYPES(IntEnum):
     UINT_8 = 1
 
 
-class PyVolume():
+class PyVolume:
     """
     Represents a volumetric dataset with an associated region of interest.
 
@@ -29,7 +29,9 @@ class PyVolume():
         data_typ (VOLUME_TYPES): The data type of the volume.
     """
 
-    def __init__(self, array: ndarray, roi: PyRegionOfIntrest, data_type: VOLUME_TYPES = ...):
+    def __init__(
+        self, np_array: ndarray, roi: PyRegionOfIntrest, data_type: VOLUME_TYPES = ...
+    ):
         """
         Initializes a PyVolume instance.
 
@@ -40,7 +42,7 @@ class PyVolume():
         """
 
         self.roi = roi
-        self.array = array
+        self.array = np_array
         self.data_typ = data_type
 
     @staticmethod
@@ -60,8 +62,8 @@ class PyVolume():
         elif volume_type == VOLUME_TYPES.UINT_8:
             return np.uint8
         else:
-            raise ValueError('Datatype is not implemented')
-        
+            raise ValueError("Datatype is not implemented")
+
     @staticmethod
     def enum_to_numpify(volume_type: VOLUME_TYPES) -> str:
         """
@@ -75,14 +77,14 @@ class PyVolume():
         """
 
         if volume_type == VOLUME_TYPES.UINT_16:
-            return 'mono16'
+            return "mono16"
         elif volume_type == VOLUME_TYPES.UINT_8:
-            return 'mono8'
+            return "mono8"
         else:
-            raise ValueError('Datatype is not implemented')
-        
+            raise ValueError("Datatype is not implemented")
+
     @classmethod
-    def dummy(cls) -> 'PyVolume':
+    def dummy(cls) -> "PyVolume":
         """
         Creates a dummy instance of PyVolume for testing.
 
@@ -91,12 +93,12 @@ class PyVolume():
         """
 
         roi = PyRegionOfIntrest.dummy()
-        array = np.random.randint(0, 255, size=roi.shape)
+        np_array = np.random.randint(0, 255, size=roi.shape)
         data_type = VOLUME_TYPES.UINT_8
-        return cls(array, roi, data_type)
+        return cls(np_array, roi, data_type)
 
     @classmethod
-    def from_message(cls, msg: Volume) -> 'PyVolume':
+    def from_message(cls, msg: Volume) -> "PyVolume":
         """
         Creates an instance of PyVolume from a ROS message.
 
@@ -108,33 +110,34 @@ class PyVolume():
         """
 
         roi: Marker = msg.grid.region_of_intrest_stack.markers[0]
-        center_points_mm = np.array([
-            roi.pose.position.x,
-            roi.pose.position.y,
-            roi.pose.position.z])
-            
-        dimensions_mm = np.array([
-            roi.scale.x,
-            roi.scale.y,
-            roi.scale.z])
-        
+        center_points_mm = np.array(
+            [roi.pose.position.x, roi.pose.position.y, roi.pose.position.z]
+        )
+
+        dimensions_mm = np.array([roi.scale.x, roi.scale.y, roi.scale.z])
+
         frame_id = roi.header.frame_id
-        resolution_mm = np.array([
-            msg.grid.resolution.x,
-            msg.grid.resolution.y,
-            msg.grid.resolution.z])
-        
-        py_roi = PyRegionOfIntrest(center_points_mm, dimensions_mm, frame_id, resolution_mm)
-    
+        resolution_mm = np.array(
+            [msg.grid.resolution.x, msg.grid.resolution.y, msg.grid.resolution.z]
+        )
+
+        py_roi = PyRegionOfIntrest(
+            center_points_mm, dimensions_mm, frame_id, resolution_mm
+        )
+
         data_typ = msg.datatype
 
-        array = np.zeros(py_roi.shape, dtype=cls.get_data_type(data_typ))
+        np_array = np.zeros(py_roi.shape, dtype=cls.get_data_type(data_typ))
 
         for i, slice in enumerate(msg.slices):
-            array[:, :, i] = ros2_numpy.numpify(slice).reshape((py_roi.shape[0], py_roi.shape[1])).astype(cls.get_data_type(data_typ))
+            np_array[:, :, i] = (
+                ros2_numpy.numpify(slice)
+                .reshape((py_roi.shape[0], py_roi.shape[1]))
+                .astype(cls.get_data_type(data_typ))
+            )
+
+        return cls(np_array, py_roi, data_typ)
 
-        return cls(array, py_roi, data_typ)
-    
     def as_message(self) -> Volume:
         """
         Converts the PyVolume instance to a ROS message.
@@ -151,12 +154,15 @@ class PyVolume():
 
         for z in range(self.array.shape[2]):
             message.slices.append(
-                ros2_numpy.msgify(Image, 
-                                  self.array[:, :, z].astype(self.get_data_type(self.data_typ)), 
-                                  self.enum_to_numpify(self.data_typ)))
-        
+                ros2_numpy.msgify(
+                    Image,
+                    self.array[:, :, z].astype(self.get_data_type(self.data_typ)),
+                    self.enum_to_numpify(self.data_typ),
+                )
+            )
+
         return message
-    
+
     @property
     def shape(self):
         """
@@ -165,10 +171,5 @@ class PyVolume():
         Returns:
             tuple: The shape of the volume.
         """
-        
-        return self.array.shape
-                                                         
-        
-
-        
 
+        return self.array.shape
diff --git a/rq_controller/rq_workflow.py b/rq_controller/rq_workflow.py
index f79cd44..d315759 100644
--- a/rq_controller/rq_workflow.py
+++ b/rq_controller/rq_workflow.py
@@ -5,55 +5,52 @@ from rq_ddetection.defect_detection_client import DefectDetectionClient
 from rq_reconstruction.reconstruction_client import ReconstructionClient
 from rq_trajectory.trajectory_optimization_client import TrajectoryOptimizationClient
 
-from rq_controller.common import PyProjection, PyRegionOfIntrest, PyVolume
+from rq_controller.common import PyProjection, PyRegionOfIntrest
 
 
 import rclpy
-from rclpy.node import Node
 
-from scipy.spatial.transform import Rotation
-import numpy as np
 
-
-class WorkflowNode():
+class WorkflowNode:
     def __init__(self):
-
         self.hardware_interface = HardwareClient()
         self.defect_detection_interface = DefectDetectionClient()
         self.trajectory = TrajectoryOptimizationClient()
         self.reconstruction = ReconstructionClient()
 
-
-    def defect_detection(self, projections: list[PyProjection]) -> list[PyRegionOfIntrest]:
+    def defect_detection(
+        self, projections: list[PyProjection]
+    ) -> list[PyRegionOfIntrest]:
         future = self.defect_detection_interface.multi_projection_defect_detection(
-            projections)
-        
+            projections
+        )
+
         rclpy.spin_until_future_complete(self.defect_detection_interface, future)
 
         response = future.result()
         roi_list = self.defect_detection_interface.response_2_py(response)
 
         return roi_list
-    
+
     def aquire_projection(self, projection: PyProjection):
         future = self.hardware_interface.aquire_projection(projection)
-        
+
         rclpy.spin_until_future_complete(self.hardware_interface, future)
 
         response = future.result()
         projection = self.hardware_interface.projection_response_2_py(response)
 
         return projection
-    
+
     def check_reachability(self, projection: PyProjection):
         future = self.hardware_interface.check_reachability(projection)
-        
+
         rclpy.spin_until_future_complete(self.hardware_interface, future)
 
         response = future.result()
         reached, cost, _ = self.hardware_interface.reachability_response_2_py(response)
         return reached, cost
-    
+
     def get_volume(self, projection_stack: list[PyProjection], roi: PyRegionOfIntrest):
         future = self.reconstruction.get_volume(projection_stack, roi)
 
@@ -61,7 +58,7 @@ class WorkflowNode():
 
         response = future.result()
         return self.reconstruction.response_2_py(response)
-    
+
     def trajectory_update(self, current_projection: PyProjection):
         future = self.trajectory.trajectory_update(current_projection)
 
@@ -69,43 +66,45 @@ class WorkflowNode():
 
         response = future.result()
         return self.trajectory.trajectory_update_response_2_py(response)
-    
+
     def trajectory_initialize(self, roi: list[PyRegionOfIntrest]):
         future = self.trajectory.trajectory_initialize(roi)
 
         rclpy.spin_until_future_complete(self.trajectory, future)
 
-    
 
 def main():
-    print('This minimal example need the nodes described in the rq_workflow/launch/echo_launch.py script.')
+    print(
+        "This minimal example need the nodes described in the rq_workflow/launch/echo_launch.py script."
+    )
     rclpy.init()
     workflow = WorkflowNode()
 
-    print('Aquire some scout views...')
+    print("Aquire some scout views...")
     projection_list = list()
     for i in range(5):
-        print(f' - At projection {i}')
+        print(f" - At projection {i}")
         scan_pose = PyProjection.dummy()
         scan_pose.focal_spot_mm += i
         projection_list.append(workflow.aquire_projection(scan_pose))
 
-    print('Detect errors in scout views ...')
+    print("Detect errors in scout views ...")
     roi = workflow.defect_detection(projection_list)
 
-    print('Init traj opt ...')
+    print("Init traj opt ...")
     workflow.trajectory_initialize(roi)
 
-    print('Get next scan pose from traj opt ...')
+    print("Get next scan pose from traj opt ...")
     next_scan_pose, finished = workflow.trajectory_update(projection_list[-1])
 
-    print('Aquire scan pose ...')
+    print("Aquire scan pose ...")
     projection_list.append(workflow.aquire_projection(next_scan_pose))
 
-    print('Resconstruct projections ...')
-    volume = workflow.get_volume(projection_list, roi[0])
+    print("Resconstruct projections ...")
+    workflow.get_volume(projection_list, roi[0])
+
+    print("FINISHED !!!")
 
-    print('FINISHED !!!')
 
-if __name__ == '__main__':
-    main()
\ No newline at end of file
+if __name__ == "__main__":
+    main()
diff --git a/rq_controller/tf2/static_broadcaster.py b/rq_controller/tf2/static_broadcaster.py
index 39f5cd9..5becb1f 100644
--- a/rq_controller/tf2/static_broadcaster.py
+++ b/rq_controller/tf2/static_broadcaster.py
@@ -1,9 +1,5 @@
-import math
-import sys
-
 from geometry_msgs.msg import TransformStamped
 
-import numpy as np
 
 import rclpy
 from rclpy.node import Node
@@ -20,13 +16,22 @@ class StaticFramePublisher(Node):
     time.
     """
 
-    def __init__(self, frame, position, quaternion, name: str = 'static_broadcaster', parent_frame: str = 'world'):
+    def __init__(
+        self,
+        frame,
+        position,
+        quaternion,
+        name: str = "static_broadcaster",
+        parent_frame: str = "world",
+    ):
         super().__init__(name)
 
         self.tf_static_broadcaster = StaticTransformBroadcaster(self)
 
         # Publish static transforms once at startup
-        self.make_transforms(frame, position.split(" "), quaternion.split(" "), parent_frame)
+        self.make_transforms(
+            frame, position.split(" "), quaternion.split(" "), parent_frame
+        )
 
     def make_transforms(self, frame, position, quaternion, parent_frame):
         t = TransformStamped()
@@ -47,27 +52,22 @@ class StaticFramePublisher(Node):
 
 
 def main(args=None):
-    logger = rclpy.logging.get_logger('logger')
+    rclpy.logging.get_logger("logger")
 
     rclpy.init(args=args)
 
-    node = rclpy.create_node('static_broadcaster')
-    parent_frame = node.declare_parameter('parent_frame', 'world').value
-    frame = node.declare_parameter('frame', 'object').value
-    name = node.declare_parameter('name', 'static_broadcaster').value
-    position = node.declare_parameter('position', '0 0 0').value
-    quaternion = node.declare_parameter('quaternion', '0 0 0 1').value
+    node = rclpy.create_node("static_broadcaster")
+    parent_frame = node.declare_parameter("parent_frame", "world").value
+    frame = node.declare_parameter("frame", "object").value
+    name = node.declare_parameter("name", "static_broadcaster").value
+    position = node.declare_parameter("position", "0 0 0").value
+    quaternion = node.declare_parameter("quaternion", "0 0 0 1").value
+
+    node = StaticFramePublisher(frame, position, quaternion, name, parent_frame)
 
-    node = StaticFramePublisher(
-        frame,
-        position,
-        quaternion,
-        name,
-        parent_frame)
-    
     try:
         rclpy.spin(node)
     except KeyboardInterrupt:
         pass
 
-    rclpy.shutdown()
\ No newline at end of file
+    rclpy.shutdown()
diff --git a/setup.py b/setup.py
index 0b679c3..37ee73e 100644
--- a/setup.py
+++ b/setup.py
@@ -1,28 +1,25 @@
 from setuptools import find_packages, setup
 
-package_name = 'rq_controller'
+package_name = "rq_controller"
 
 setup(
     name=package_name,
-    version='0.0.0',
-    packages=find_packages(exclude=['test']),
+    version="0.0.0",
+    packages=find_packages(exclude=["test"]),
     data_files=[
-        ('share/ament_index/resource_index/packages',
-            ['resource/' + package_name]),
-        ('share/' + package_name, ['package.xml']),
+        ("share/ament_index/resource_index/packages", ["resource/" + package_name]),
+        ("share/" + package_name, ["package.xml"]),
     ],
-    install_requires=['setuptools',
-                      'Pillow',
-                      'pyometiff'],
+    install_requires=["setuptools", "Pillow", "pyometiff"],
     zip_safe=True,
-    maintainer='root',
-    maintainer_email='simon.wittl@th-deg.de',
-    description='TODO: Package description',
-    license='TODO: License declaration',
-    tests_require=['pytest'],
+    maintainer="root",
+    maintainer_email="simon.wittl@th-deg.de",
+    description="TODO: Package description",
+    license="TODO: License declaration",
+    tests_require=["pytest"],
     entry_points={
-        'console_scripts': [
-            'tf_static_broadcaster = rq_controller.tf2.static_broadcaster:main',
+        "console_scripts": [
+            "tf_static_broadcaster = rq_controller.tf2.static_broadcaster:main",
         ],
     },
 )
diff --git a/test/__pycache__/test_base_loader.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_base_loader.cpython-38-pytest-8.2.2.pyc
index 279c0db1aec0f0f5b456424894890499d32a6041..49fe03382c5c64fcf5b5890c4cdc5cd6c09d5e4d 100644
GIT binary patch
delta 82
zcmbO(HeHN2l$V!_0R(EJ(ywpioyExLJb4A9A0yl5?~Dgo7zH-(WYcD36rcQ*eedKv
jPQJ+-IQB6rPR{4t%BVJ3mFp=lGYcaZCm%Z>D+mGrqe&Fr

delta 80
zcmbO(HeHN2l$V!_0SKOoPDnekk#`m&WAfw`jDC!)o4+$2WMSmnypv6vkx^vwQ}(@#
i5|bBmoMx1noX@$HQDL$w*Hd057Dg@(J~loU5Cj00cod5O

diff --git a/test/__pycache__/test_base_writer.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_base_writer.cpython-38-pytest-8.2.2.pyc
index 29e343a89d1a629bacda3ebb8b0f24f419a5fc0e..1d58ab39dfe5e08797f2dc9b57d4f0e5bc24e22c 100644
GIT binary patch
delta 262
zcmewr)E3Ma%FD~e00Om9>AM9t@@X<LI#0G>@?+%L+|IO%g;8v?6T2g$uq;qTF<2oZ
z4<idB8w1<_$rCsZGpbAu;k+xXiAC0qYd538=3iWgm>5kb-xrtNe2aHJ6QlLyd{NoW
z(*;tPCeM<Q+bk&@%EaioIbSrNi7{~VeepCV#z?S=WGQnd#ssjw9WvTXjA>v|M)@Wt
z#_Y-WRpmD?R%B*kECe%;DN8UhmT$hVD#pZE3l<U8uwztb<6&lD<l^LD2SQdbW&_fk
N93a5X0b;X(AON|#JYE0*

delta 256
zcmZn*{uRU*%FD~e00dXXCZv53*vO~J#F#wUg2|7OZF4)*E*3`q%}(r&jQpZNCB-Zt
zg^WCmMS7EcxY8#7<A`RIoE*h@msb{9cCr`OZbrq;Y}|*K7*!{~7MI=pjCVd0qt@gP
za<ZEj3#2eH>VZX7g+rMbjlrTbM1z<Z%{RXmPh(=V0gL2InKMmh7na+6L`Iv5(RK3&
zIiL}qn<pqLF){j0o}ehd`I52(6Js!#$))z4kud_yRMfCzRA=O2VqxUs;9%omfkIXw
Jjfz=0SOL)zJTw3R

diff --git a/test/__pycache__/test_projection.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_projection.cpython-38-pytest-8.2.2.pyc
index 8413a88371ec487791bfb98a4279f60cf8b3edae..cb9728ada1db97c0497e0512d7c4c04df1beda91 100644
GIT binary patch
delta 3507
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

delta 4107
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
C4A|5F

diff --git a/test/__pycache__/test_projection_geometry.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_projection_geometry.cpython-38-pytest-8.2.2.pyc
index daecbc751bbc7609edc20368ee278eb37f545dfe..e30f6eb8b6c80bf2b52bcc7c18ee64568b1a801f 100644
GIT binary patch
delta 450
zcmYk(%}WAN5CHI<^=s8!O;REy1=)5b?ZfOtDoaf*^6R1pk)lIHbja2q45AK2f4~Uc
zJ9m-2)v0TT&an^$o}yFdF1;~`-oWq8do#@8y|>0^L##R+HjdU=ea|X(#X1ieZJ+N6
zpkvWK74oxEYl|Ad#lj6naPvI;#kWzwj(y%~FmnQD=9=69Cvs{`)9sHYi*9VTr(?Qy
zJ9uY+IF>zmFyM_R1bOYj^KPuou(7I#@Dg#xheE`NCAwQ6fR2QkAXW%7F-j~F{pMxL
zlK3@p8Awr@CNjhtF-c^J9CmyXr19Q&02OV_-?hRzzDOs&B&(i^JFV8PdK{FLW;+pX
z=?1o`8~7>}@k#b#Ps*dDII$SE;FX-kL)i%hJeTWGWNvDbq5zb%QaH|=%50vF*W(vK
z-M-pvZ~so7`+t~KbNwbc5`}_Rjrag+coc204|tBIT=W!bo~RHt3``^}L&C5i3br5L
Ce`;|6

delta 482
zcmZ9}OG`pQ6ae6RuCHt6Bg2Xelft|=--?!LrD<822vIF=LP!)u8@aYo2wJo+IDbHB
z*#qSMf;ROJg4`4cf%F@K=Gdm1fp5;7b7p2SpOv?YW!G#T;Mll-y*j=&w=FUchw2I6
z6+p(4DLm?AwfZ)7fQ?N%Xu;0gcx;*Gtf1!vPS4f2LC(S{5%t6Ls58jgS|bvb)rX-w
z4b0-cqX2+6j%moN&yIKPUX)#>IEWD<#^{Lz6?$pP#R;G-p(KcYK~Is3m?x6N0+GV6
z@yp323YUotktIUJ3X#K>(*rSla-Kp-jZU<U=2gnmeG)Wxuqp-7HM2OCVu_}uTCX1|
zXKs&JYb2$*tYM3?_P2EmU%XbK@IUGHT45c-UKuu+MQVagiqwcG0*1T>EQm$*R_cOo
ziyf8`eHU)ol&LiieqK#<er@7PuHO{%2OzCh{Z4=~o(C%IgF1n%Ey9XQo+uGz^aYcK
KVZkaG1>-lW*lqU!

diff --git a/test/__pycache__/test_region_of_intrest.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_region_of_intrest.cpython-38-pytest-8.2.2.pyc
index 469b57f673dd3002e757812d9c29a8ea600abac0..cdd2ab3fba79761b42e2adb2ccc1cc222fb2eecd 100644
GIT binary patch
delta 357
zcmbQ1G%blQl$V!_0R(EJ(z*FI@})5`I!`WT3TEWme3HqVol#-4C|48<qu%BMzFF*y
zW}CUiQg|6}O)gM2Vzi&USotnv{N_d#1xBFIBGm@Q;K>`*r5TeZH>&KJ{9A1*W8&nA
z>T-+`VA1r=XVm+cq>_Pl6{UcPR1jegA~Zon+U621W=4@)ECrQ@#+uA8{{ul$7Dy&*
za<{e;(1vx|L5vxb8Ffs6Y%?88#@NXPI)<XmYz!O>Yz$mLk_$+;G4L@I<!;`lGmlX;
z53CzR*nx<A5K%BWQ|~&^07rcb(Oc~4MVTq_xw*GE3i31aN`S1QlF5DgR*YGbPwFc(
zR!@GbAHx_m*~`EPsH4`vN;Dp1L=uQd2NAg-qI&Wf17mS!9!4H!7Dg^n4n7VJ4o(gc
J4pt!K1ptOoV2l6&

delta 357
zcmbQ1G%blQl$V!_0SFQv^`|}J-N={5#F#v}lqr~zee+2sZ+1qR&7xdUER33)3;1TS
zGa78>7E9q}jGA1aY{Y0Wd9m_c#<<OmDhiB1p+%|<jNX$ss7o^@Om0-!Gx@jLRL1zp
z6V>Gy{lTKCo6o5CF-au?Eh|a_5y>FJ9z<w@h?LDGTFi_hQ7i?OhQ^xAFaHBUQ94K_
zeR8+963~Wq+ChwIlNohPfNV1zOUB^I1v-YJjBE@X3~UTsK#~hcw=wWB6lHAQrZbOG
zG!v{FMA(6dED(`BIaBXC&;Un$3(+X{^rFm^_}ttmj)MHmyb>U*D1UOFz7=Ei<dgc!
zjK!1R>c=qNn(Sp@1k_P$U?my{G9m#)q=JYH5K%n&jDfK@BM&1F6AL4k2nR0*I|m1c
KFb4||asUADj$PUS

diff --git a/test/__pycache__/test_volume.cpython-38-pytest-8.2.2.pyc b/test/__pycache__/test_volume.cpython-38-pytest-8.2.2.pyc
index e4329085a29c878eed2392f2e4b4eda540f9cb6f..f39dd0ea1a983260f04c5ddbbe585720a430ed67 100644
GIT binary patch
delta 2064
zcmb7FOK%%h6!x{9di;tV6T9R=;y8)Igpk(G11C+Jv}s78rKAx~ThnGb#S=5aFYb(`
zI3@CmN(fW|S8TdK?XnvX@{$cJkl1vEght))8(<Y7!Ts(;$vA4IG?6~vIrp41=R2=?
zw)$)}oC}8oF8uB0i$DAn{@qP1_NP1Tv%E#B7c3U_>~hHi&l~+RSF(tn_v>Y~s3~Qm
z_{2pgxo|JL%>M9PBAsl&n<l4QYu>-zB*iZJs$<jqn$Ez`Q)^7`bgG7-Q8PLEdaW@l
z8#K#ro{*g@?5|)S+l)@GX9K&`3!#@Dn9{Rw0@XdIeU{tLaiN@`9{yu>6zb)2gQlQ7
z3>X1Su(~u(rr0B?Miy9?e>k`R@f0A#7X76WFc388DSonEF*U<f^onlkYDwQwO}$!~
zU{C#L$VMv|_?eJ-_FM3Ne}?PIC!QB1onv}vsCf=<L4_^?mH;B$D0Sm}9w5S)z)3{4
ziqjRq3gB(PCBQp?cLA3H?*WAGWqy(a6w%PDfOWu{%N2E5!KGS5u@gQ=bF3$vjDLXJ
zYXAppb~T(JvrG%0BDY)Lg~#2T;xCc4n4s8J2bPMOK?fc7IQmKRki*M|pF}8vdM`h5
zBBvo}XXE5COLJVomTXmNSv9ll%dW@^c>UOwAZOXHU5lj9IuZK<yuR+v#mDiEqm^NM
z-FM<>IW%gT&O+xcz#OZ_<K#N~JbtT*-m-#5N&iSwR7%x`+~MeeLz_*m$J<iW`3`IX
zs-cuM!{Bp6hq%CgeG;eRCXG*D(?vCS^J|rQS);0{(J?&x_+w4q+VA~~@QR|yU5<1L
z(xO9<LAy;3y1|*^41FI8yoFp;1$+p2lS6*SZ&T-!Pm7S=1vom1KUxtnKZkr*Sog{E
zq}19txr_ol?0GbbG|IjgW{xWG!<61c!1D59Ovz;DEB3f&lPwKKSfsa~Nqv!ZD_qn}
zrJ$OsVm4};BcpInM*+vpEvATGV0vFFB2pwW0P+@l*w?#cg)m8Wzoj#6Pt!-~#{~Vh
z(5|t6`ew*>YrOwoBF%HJoR7i2iBGc5!3_JaC(6Dbh-MDgU3(wV>j>@!;3gvKC|4`h
zEO$ZoU95zv@M*6Vp0>e6GYRdNBL68YD8S&ri_@M5n#WZgAll&;PDcvgEtsP<LyB2d
z_IvWix5*iDuQi(tkYE~CVZc1QJhavnb1v40W2f+IIVLu`y&^?59s3;S7XWU+!4f<!
zU0JgC4S($!=cakiR2(H35Cn~n)taWTTxu}(8Dbr<q`P&!Wa<@zuCV*5sf0*85^8lI
z3HDg#+0)cb{vnlz`<qt8&?-ii;uC62Vw!y5G6oahmcDJ)sb-Jv2IRK^1waX~4Y&uu
z`yD}wdWdhBv{0@V>Lu+Q;TpyW2gwz4Nkk%1Z-~4aBKVg)UJ`gs3P?Ull0qDP{{zJW
BwnhK|

delta 2064
zcmb7FOK%%h6!wjsdi<&%#4e80*l`=kBqhN~Y`IOzqYqMOiP97oGK4tOW|9u^OLxW+
zj#PO?B?N-P6`L+lyFuy-Aurjm0*OsmNNCgz`~_G=mEeANqGS@aQkun=d(XZ1%=ymw
z&hgXvr}LpwDCBqXzrUY-z460Oq2IfR!v5^ac@@tl)z=jkao^>ZuiPVjgZ!37bj7Dv
z)lIEjH8vl+=rA|#6mPRX+*e3|^?P#UOlQ&aPZvqEOWykHZ}MZB!$WtYHMxCUH4KfK
znZcn(s~{V6lAk;#+t=7%ffQSfj4l=Zcc}+P58Zo8&%z0=?l|8U`TIOK$_eV`|E%uy
zX0_U)X<W_#1^`2>DGie`_E2h&B8&P?1&T0_0Y=%3uW}j;1PwaESJo@0W|(EYrklE2
z(YIAouh(Su#5YBjIsyOBgcR6sfqQ+U{H}cLc|p=V(}M%;b8rhPbP8}DAi|B%F09`I
zh%kn*5>Z{m>Iz^EZ~<@$a2fD6U>@)eK=_{FE7?yG4Sg4|0J!FIMO;>3w$Uov2|r7(
zvt%d}U&QHq00(Qf5K52<riI4Ha_76ya2KceOL#FRC?2Q-%SOzAgARKX*=p}|c=_;8
zA{0SA$yc1n85j<-aq=k8>->Uk*{IX1Y8KfS(eMj+{TNM<bL`jX3|Z?Oi+v7W_j^k5
z*YJ(Qm0>$QYjH?U;X=*Paom{z<XJr)CpXw<@s&2TWd)3iep@T6l&USc+o6E{HwwSU
zWhv^sf;WEEC|5PZ;C(~;;Z=NIus>?jc=xqkRE0O6R%=!@s+t--&G(9zZyWl?uJ&i)
zA4QR)j&usrLLtbYEt5SpC=RKN)2q0^CFG(X13m!A9F)Azrp`N`-hy=%;7}6ZT4B*Y
z`+P63o)dFK>)bqX5e0aVd^m_S%H9`dj()_?6;|#|_!SQ?#)M3^zhsY+t86wMX5rpG
zCZ)nlN@!Cv%j>GCmd#c}b7WLZfGk2gYHra*beid@Y*?g7WB}w9_8`?es|3+Ww%&5%
z2X>Q(>Bj{91EDRje^Ys~)fw)4Mx+Aw%J~@VoA@X@o*rlaB_r(n{>b?LxI3sLdIQ0|
z4=5p`?rObOFLD=j*Tv=F27KCMg{Li@Xup)O1o=()zXQ<W%xe#my*$vHIQj%2<nSR@
zhYH^#m_wSuvRN<hYO?fgCQmAzLdH)5Z{Sr3P+(UF7TcoF#rSZ{6n-qn#6-78q^M>F
zkL-Dl)igkK*WM63YOQ6-o;UonQ_e~AoUS-b&@TuYchrWq!b;h6tjfcX`z`5KQ?HnM
z&7gDaUUniORELCG-AIDnmKJ-GUFJWe@~OVI5;nA&QK$F`HAc}*UJybrTiy-*Q?p4m
zyLFdfUk0oJG{6?10>JkjL5g~eaGNEauh!R_6>XaEodLg+E9R1jL?R?eUJY^x63Oi$
P{*#hl@=B5v<mmk$#$u~_

diff --git a/test/test_base_loader.py b/test/test_base_loader.py
index eb1c071..d00fcbc 100644
--- a/test/test_base_loader.py
+++ b/test/test_base_loader.py
@@ -2,33 +2,39 @@ import pytest
 from rq_controller.common.io.loader import BaseDataLoader
 from pathlib import Path
 
+
 @pytest.fixture
 def base_data_loader():
     return BaseDataLoader(
         porjection_geometry_suffix="geometry_suffix",
         projection_suffix="projection_suffix",
         region_of_intrest_suffix="roi_suffix",
-        volume_suffix="volume_suffix"
+        volume_suffix="volume_suffix",
     )
 
+
 def test_initialization(base_data_loader: BaseDataLoader):
     assert base_data_loader.porjection_geometry_suffix == "geometry_suffix"
     assert base_data_loader.projection_suffix == "projection_suffix"
     assert base_data_loader.region_of_intrest_suffix == "roi_suffix"
     assert base_data_loader.volume_suffix == "volume_suffix"
 
+
 def test_load_projection_geometry_not_implemented(base_data_loader: BaseDataLoader):
     with pytest.raises(NotImplementedError):
         base_data_loader.load_projection_geometry(Path("/fake/path"))
 
+
 def test_load_projection_not_implemented(base_data_loader: BaseDataLoader):
     with pytest.raises(NotImplementedError):
         base_data_loader.load_projection(Path("/fake/path"))
 
+
 def test_load_region_of_intrest_not_implemented(base_data_loader: BaseDataLoader):
     with pytest.raises(NotImplementedError):
         base_data_loader.load_region_of_intrest(Path("/fake/path"))
 
+
 def test_load_volume_not_implemented(base_data_loader: BaseDataLoader):
     with pytest.raises(NotImplementedError):
-        base_data_loader.load_volume(Path("/fake/path"))
\ No newline at end of file
+        base_data_loader.load_volume(Path("/fake/path"))
diff --git a/test/test_base_writer.py b/test/test_base_writer.py
index 3d41a2d..d0896df 100644
--- a/test/test_base_writer.py
+++ b/test/test_base_writer.py
@@ -1,91 +1,128 @@
 import pytest
 from pathlib import Path
-from rq_controller.common import PyProjection, PyProjectionGeometry, PyRegionOfIntrest, PyVolume
+from rq_controller.common import (
+    PyProjection,
+    PyProjectionGeometry,
+    PyRegionOfIntrest,
+    PyVolume,
+)
 from rq_controller.common.io.writer import BaseDataWriter
 
+
 @pytest.fixture
 def base_data_writer():
     return BaseDataWriter(
         porjection_geometry_suffix=".geom",
         projection_suffix=".proj",
         region_of_intrest_suffix=".roi",
-        volume_suffix=".vol"
+        volume_suffix=".vol",
     )
 
+
 def test_initialization(base_data_writer):
     assert base_data_writer.porjection_geometry_suffix == ".geom"
     assert base_data_writer.projection_suffix == ".proj"
     assert base_data_writer.region_of_intrest_suffix == ".roi"
     assert base_data_writer.volume_suffix == ".vol"
 
+
 def test_write_projection_geometry_not_implemented(base_data_writer):
     with pytest.raises(NotImplementedError):
-        base_data_writer.write_projection_geometry(Path("/fake/path"), PyProjectionGeometry.dummy())
+        base_data_writer.write_projection_geometry(
+            Path("/fake/path"), PyProjectionGeometry.dummy()
+        )
+
 
 def test_write_projection_not_implemented(base_data_writer):
     with pytest.raises(NotImplementedError):
         base_data_writer.write_projection(Path("/fake/path"), PyProjection.dummy())
 
+
 def test_write_region_of_intrest_not_implemented(base_data_writer):
     with pytest.raises(NotImplementedError):
-        base_data_writer.write_region_of_intrest(Path("/fake/path"), PyRegionOfIntrest.dummy())
+        base_data_writer.write_region_of_intrest(
+            Path("/fake/path"), PyRegionOfIntrest.dummy()
+        )
+
 
 def test_write_volume_not_implemented(base_data_writer):
     with pytest.raises(NotImplementedError):
         base_data_writer.write_volume(Path("/fake/path"), PyVolume.dummy())
 
+
 def test_get_next_projection_save_path(base_data_writer, tmp_path):
     (tmp_path / "projection_00001.proj").touch()
     expected_path = tmp_path / "projection_00002.proj"
     assert base_data_writer.get_next_projection_save_path(tmp_path) == expected_path
 
+
 def test_get_projection_save_path_i(base_data_writer, tmp_path):
     expected_path = tmp_path / "projection_00005.proj"
     assert base_data_writer.get_projection_save_path_i(tmp_path, 5) == expected_path
 
+
 def test_get_next_projection_geometry_save_path(base_data_writer, tmp_path):
     (tmp_path / "geometry_00001.geom").touch()
     expected_path = tmp_path / "geometry_00002.geom"
-    assert base_data_writer.get_next_projection_geometry_save_path(tmp_path) == expected_path
+    assert (
+        base_data_writer.get_next_projection_geometry_save_path(tmp_path)
+        == expected_path
+    )
+
 
 def test_get_projection_geometry_save_path_i(base_data_writer, tmp_path):
     expected_path = tmp_path / "geometry_00005.geom"
-    assert base_data_writer.get_projection_geometry_save_path_i(tmp_path, 5) == expected_path
+    assert (
+        base_data_writer.get_projection_geometry_save_path_i(tmp_path, 5)
+        == expected_path
+    )
+
 
 def test_get_next_region_of_intrest_save_path(base_data_writer, tmp_path):
     (tmp_path / "roi_00001.roi").touch()
     expected_path = tmp_path / "roi_00002.roi"
-    assert base_data_writer.get_next_region_of_intrest_save_path(tmp_path) == expected_path
+    assert (
+        base_data_writer.get_next_region_of_intrest_save_path(tmp_path) == expected_path
+    )
+
 
 def test_get_region_of_intrest_save_path_i(base_data_writer, tmp_path):
     expected_path = tmp_path / "roi_00005.roi"
-    assert base_data_writer.get_region_of_intrest_save_path_i(tmp_path, 5) == expected_path
+    assert (
+        base_data_writer.get_region_of_intrest_save_path_i(tmp_path, 5) == expected_path
+    )
+
 
 def test_get_next_volume_save_path(base_data_writer, tmp_path):
     (tmp_path / "volume_00001.vol").touch()
     expected_path = tmp_path / "volume_00002.vol"
     assert base_data_writer.get_next_volume_save_path(tmp_path) == expected_path
 
+
 def test_get_volume_save_path_i(base_data_writer, tmp_path):
     expected_path = tmp_path / "volume_00005.vol"
     assert base_data_writer.get_volume_save_path_i(tmp_path, 5) == expected_path
 
+
 def test_number_of_projection_geometries(base_data_writer, tmp_path):
     (tmp_path / "geometry_00001.geom").touch()
     (tmp_path / "geometry_00002.geom").touch()
     assert base_data_writer.number_of_projection_geometries(tmp_path) == 2
 
+
 def test_number_of_projections(base_data_writer, tmp_path):
     (tmp_path / "projection_00001.proj").touch()
     (tmp_path / "projection_00002.proj").touch()
     assert base_data_writer.number_of_projections(tmp_path) == 2
 
+
 def test_number_of_region_of_intrests(base_data_writer, tmp_path):
     (tmp_path / "roi_00001.roi").touch()
     (tmp_path / "roi_00002.roi").touch()
     assert base_data_writer.number_of_region_of_intrests(tmp_path) == 2
 
+
 def test_number_of_volumes(base_data_writer, tmp_path):
     (tmp_path / "volume_00001.vol").touch()
     (tmp_path / "volume_00002.vol").touch()
-    assert base_data_writer.number_of_volumes(tmp_path) == 2
\ No newline at end of file
+    assert base_data_writer.number_of_volumes(tmp_path) == 2
diff --git a/test/test_projection.py b/test/test_projection.py
index 06075e4..0ccf386 100644
--- a/test/test_projection.py
+++ b/test/test_projection.py
@@ -1,10 +1,13 @@
 import pytest
 import numpy as np
-from rq_controller.common import PyProjection  # Adjust this import according to your module's path
+from rq_controller.common import (
+    PyProjection,
+)  # Adjust this import according to your module's path
 from rq_interfaces.msg import Projection
 from sensor_msgs.msg import Image
 import ros2_numpy
 
+
 @pytest.fixture
 def example_message():
     msg = Projection()
@@ -23,10 +26,10 @@ def example_message():
     msg.projection_geometry.focal_spot_orientation_quad.z = 0.0
     msg.projection_geometry.focal_spot_orientation_quad.w = 1.0
     msg.projection_geometry.header.frame_id = "test_frame"
-    
+
     image_array = np.zeros((10, 10), dtype=np.uint16)
-    msg.image = ros2_numpy.msgify(Image, image_array, encoding='mono16')
-    
+    msg.image = ros2_numpy.msgify(Image, image_array, encoding="mono16")
+
     msg.detector_heigth_mm = 100.0
     msg.detector_width_mm = 200.0
     msg.voltage_kv = 120.0
@@ -34,6 +37,7 @@ def example_message():
     msg.exposure_time_ms = 500.0
     return msg
 
+
 def test_initialization():
     focal_spot_mm = np.array([1.0, 2.0, 3.0])
     detector_postion_mm = np.array([4.0, 5.0, 6.0])
@@ -47,12 +51,28 @@ def test_initialization():
     frame_id = "test_frame"
     focal_spot_orientation_quad = np.array([0.0, 0.0, 0.0, 1.0])
 
-    projection = PyProjection(focal_spot_mm, detector_postion_mm, detector_orientation_quad, image, detector_heigth_mm, detector_width_mm, frame_id, focal_spot_orientation_quad, voltage_kv, current_ua, exposure_time_ms)
+    projection = PyProjection(
+        focal_spot_mm,
+        detector_postion_mm,
+        detector_orientation_quad,
+        image,
+        detector_heigth_mm,
+        detector_width_mm,
+        frame_id,
+        focal_spot_orientation_quad,
+        voltage_kv,
+        current_ua,
+        exposure_time_ms,
+    )
 
     assert np.array_equal(projection.focal_spot_mm, focal_spot_mm)
     assert np.array_equal(projection.detector_postion_mm, detector_postion_mm)
-    assert np.array_equal(projection.detector_orientation_quad, detector_orientation_quad)
-    assert np.array_equal(projection.focal_spot_orientation_quad, focal_spot_orientation_quad)
+    assert np.array_equal(
+        projection.detector_orientation_quad, detector_orientation_quad
+    )
+    assert np.array_equal(
+        projection.focal_spot_orientation_quad, focal_spot_orientation_quad
+    )
     assert np.array_equal(projection.image, image)
     assert projection.detector_heigth_mm == detector_heigth_mm
     assert projection.detector_width_mm == detector_width_mm
@@ -61,25 +81,32 @@ def test_initialization():
     assert projection.exposure_time_ms == exposure_time_ms
     assert projection.frame_id == frame_id
 
+
 def test_dummy_method():
     projection = PyProjection.dummy()
 
     assert np.array_equal(projection.focal_spot_mm, np.array([0.0, 100.0, 0.0]))
     assert np.array_equal(projection.detector_postion_mm, np.array([0.0, -100.0, 0.0]))
-    assert np.array_equal(projection.detector_orientation_quad, np.array([1.0, 0.0, 0.0, 1.0]))
-    assert np.array_equal(projection.image, np.zeros((10, 10), dtype=np.uint16))
+    assert np.array_equal(
+        projection.detector_orientation_quad, np.array([1.0, 0.0, 0.0, 1.0])
+    )
     assert projection.detector_heigth_mm == 10.0
     assert projection.detector_width_mm == 10.0
     assert projection.frame_id == "object"
 
+
 def test_from_message(example_message):
     projection = PyProjection.from_message(example_message)
 
     assert np.array_equal(projection.focal_spot_mm, np.array([10.0, 20.0, 30.0]))
     assert np.array_equal(projection.detector_postion_mm, np.array([40.0, 50.0, 60.0]))
-    assert np.array_equal(projection.detector_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0]))
-    assert np.array_equal(projection.focal_spot_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0]))
-    assert np.array_equal(projection.image, np.zeros((10, 10), dtype=np.uint16))
+    assert np.array_equal(
+        projection.detector_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0])
+    )
+    assert np.array_equal(
+        projection.focal_spot_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0])
+    )
+
     assert projection.detector_heigth_mm == 100.0
     assert projection.detector_width_mm == 200.0
     assert projection.voltage_kv == 120.0
@@ -87,6 +114,7 @@ def test_from_message(example_message):
     assert projection.exposure_time_ms == 500.0
     assert projection.frame_id == "test_frame"
 
+
 def test_as_message(example_message):
     projection = PyProjection.from_message(example_message)
     msg = projection.as_message()
@@ -106,13 +134,16 @@ def test_as_message(example_message):
     assert msg.projection_geometry.focal_spot_orientation_quad.z == 0.0
     assert msg.projection_geometry.focal_spot_orientation_quad.w == 1.0
     assert msg.projection_geometry.header.frame_id == "test_frame"
-    assert np.array_equal(ros2_numpy.numpify(msg.image), np.zeros((10, 10), dtype=np.uint16))
+    assert np.array_equal(
+        ros2_numpy.numpify(msg.image), np.zeros((10, 10), dtype=np.uint16)
+    )
     assert msg.detector_heigth_mm == 100.0
     assert msg.detector_width_mm == 200.0
     assert msg.voltage_kv == 120.0
     assert msg.current_ua == 150.0
     assert msg.exposure_time_ms == 500.0
 
+
 def test_properties():
     focal_spot_mm = np.array([1.0, 2.0, 3.0])
     detector_postion_mm = np.array([4.0, 5.0, 6.0])
@@ -126,9 +157,21 @@ def test_properties():
     frame_id = "test_frame"
     focal_spot_orientation_quad = np.array([0.0, 0.0, 0.0, 1.0])
 
-    projection = PyProjection(focal_spot_mm, detector_postion_mm, detector_orientation_quad, image, detector_heigth_mm, detector_width_mm, frame_id, focal_spot_orientation_quad, voltage_kv, current_ua, exposure_time_ms)
+    projection = PyProjection(
+        focal_spot_mm,
+        detector_postion_mm,
+        detector_orientation_quad,
+        image,
+        detector_heigth_mm,
+        detector_width_mm,
+        frame_id,
+        focal_spot_orientation_quad,
+        voltage_kv,
+        current_ua,
+        exposure_time_ms,
+    )
 
     assert projection.detector_heigth_px == 20
     assert projection.detector_width_px == 30
     assert projection.pixel_pitch_x_mm == 200.0 / 30
-    assert projection.pixel_pitch_y_mm == 100.0 / 20
\ No newline at end of file
+    assert projection.pixel_pitch_y_mm == 100.0 / 20
diff --git a/test/test_projection_geometry.py b/test/test_projection_geometry.py
index 7d4c44d..e28d1d9 100644
--- a/test/test_projection_geometry.py
+++ b/test/test_projection_geometry.py
@@ -1,29 +1,30 @@
 import pytest
 import numpy as np
 from rq_interfaces.msg import ProjectionGeometry
-from rq_controller.common import PyProjectionGeometry  
+from rq_controller.common import PyProjectionGeometry
 
 
 @pytest.fixture
 def example_message():
     msg = ProjectionGeometry()
-    msg.focal_spot_postion_mm.x = 10.
-    msg.focal_spot_postion_mm.y = 20.
-    msg.focal_spot_postion_mm.z = 30.
-    msg.detector_postion_mm.x = 40.
-    msg.detector_postion_mm.y = 50.
-    msg.detector_postion_mm.z = 60.
-    msg.detector_orientation_quad.x = 0.
-    msg.detector_orientation_quad.y = 0.
-    msg.detector_orientation_quad.z = 0.
-    msg.detector_orientation_quad.w = 1.
-    msg.focal_spot_orientation_quad.x = 0.
-    msg.focal_spot_orientation_quad.y = 0.
-    msg.focal_spot_orientation_quad.z = 0.
-    msg.focal_spot_orientation_quad.w = 1.
+    msg.focal_spot_postion_mm.x = 10.0
+    msg.focal_spot_postion_mm.y = 20.0
+    msg.focal_spot_postion_mm.z = 30.0
+    msg.detector_postion_mm.x = 40.0
+    msg.detector_postion_mm.y = 50.0
+    msg.detector_postion_mm.z = 60.0
+    msg.detector_orientation_quad.x = 0.0
+    msg.detector_orientation_quad.y = 0.0
+    msg.detector_orientation_quad.z = 0.0
+    msg.detector_orientation_quad.w = 1.0
+    msg.focal_spot_orientation_quad.x = 0.0
+    msg.focal_spot_orientation_quad.y = 0.0
+    msg.focal_spot_orientation_quad.z = 0.0
+    msg.focal_spot_orientation_quad.w = 1.0
     msg.header.frame_id = "test_frame"
     return msg
 
+
 def test_initialization():
     focal_spot_mm = np.array([1.0, 2.0, 3.0])
     detector_postion_mm = np.array([4.0, 5.0, 6.0])
@@ -31,22 +32,36 @@ def test_initialization():
     frame_id = "test_frame"
     focal_spot_orientation_quad = np.array([0.0, 0.0, 0.0, 1.0])
 
-    geometry = PyProjectionGeometry(focal_spot_mm, detector_postion_mm, detector_orientation_quad, frame_id, focal_spot_orientation_quad)
+    geometry = PyProjectionGeometry(
+        focal_spot_mm,
+        detector_postion_mm,
+        detector_orientation_quad,
+        frame_id,
+        focal_spot_orientation_quad,
+    )
 
     assert np.array_equal(geometry.focal_spot_mm, focal_spot_mm)
     assert np.array_equal(geometry.detector_postion_mm, detector_postion_mm)
     assert np.array_equal(geometry.detector_orientation_quad, detector_orientation_quad)
-    assert np.array_equal(geometry.focal_spot_orientation_quad, focal_spot_orientation_quad)
+    assert np.array_equal(
+        geometry.focal_spot_orientation_quad, focal_spot_orientation_quad
+    )
     assert geometry.frame_id == frame_id
 
+
 def test_dummy_method():
     geometry = PyProjectionGeometry.dummy()
 
     assert np.array_equal(geometry.focal_spot_mm, np.array([1.0, 0.0, 0.0]))
     assert np.array_equal(geometry.detector_postion_mm, np.array([-1.0, 0.0, 0.0]))
-    assert np.array_equal(geometry.detector_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0]))
+    assert np.array_equal(
+        geometry.detector_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0])
+    )
     assert geometry.frame_id == "object"
-    assert np.array_equal(geometry.focal_spot_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0]))
+    assert np.array_equal(
+        geometry.focal_spot_orientation_quad, np.array([0.0, 0.0, 0.0, 1.0])
+    )
+
 
 def test_from_message(example_message):
     geometry = PyProjectionGeometry.from_message(example_message)
@@ -57,6 +72,7 @@ def test_from_message(example_message):
     assert np.array_equal(geometry.focal_spot_orientation_quad, np.array([0, 0, 0, 1]))
     assert geometry.frame_id == "test_frame"
 
+
 def test_as_message(example_message):
     geometry = PyProjectionGeometry.from_message(example_message)
     msg = geometry.as_message()
diff --git a/test/test_region_of_intrest.py b/test/test_region_of_intrest.py
index 678f052..6774075 100644
--- a/test/test_region_of_intrest.py
+++ b/test/test_region_of_intrest.py
@@ -1,13 +1,16 @@
 import pytest
 import numpy as np
-from rq_controller.common import PyRegionOfIntrest  # Adjust this import according to your module's path
+from rq_controller.common import (
+    PyRegionOfIntrest,
+)  # Adjust this import according to your module's path
 from rq_interfaces.msg import RegionOfIntrest
 from visualization_msgs.msg import Marker
 
+
 @pytest.fixture
 def example_message():
     msg = RegionOfIntrest()
-    
+
     marker = Marker()
     marker.pose.position.x = 10.0
     marker.pose.position.y = 20.0
@@ -16,15 +19,16 @@ def example_message():
     marker.scale.y = 50.0
     marker.scale.z = 60.0
     marker.header.frame_id = "test_frame"
-    
+
     msg.region_of_intrest_stack.markers.append(marker)
-    
+
     msg.resolution.x = 0.1
     msg.resolution.y = 0.1
     msg.resolution.z = 0.1
-    
+
     return msg
 
+
 def test_initialization():
     center_points_mm = np.array([[1.0, 2.0, 3.0]])
     dimensions_mm = np.array([[4.0, 5.0, 6.0]])
@@ -38,6 +42,7 @@ def test_initialization():
     assert np.array_equal(roi.resolution_mm, resolution_mm)
     assert roi.frame_id == frame_id
 
+
 def test_dummy_method():
     roi = PyRegionOfIntrest.dummy()
 
@@ -45,6 +50,7 @@ def test_dummy_method():
     assert roi.dimensions_mm.shape == (1, 3)
     assert roi.frame_id == "object"
 
+
 def test_from_message(example_message):
     roi = PyRegionOfIntrest.from_message(example_message)
 
@@ -53,6 +59,7 @@ def test_from_message(example_message):
     assert np.array_equal(roi.resolution_mm, np.array([[0.1, 0.1, 0.1]]))
     assert roi.frame_id == "test_frame"
 
+
 def test_as_message(example_message):
     roi = PyRegionOfIntrest.from_message(example_message)
     msg = roi.as_message()
@@ -67,26 +74,34 @@ def test_as_message(example_message):
     assert marker.scale.y == 50.0
     assert marker.scale.z == 60.0
     assert marker.header.frame_id == "test_frame"
-    
+
     assert msg.resolution.x == 0.1
     assert msg.resolution.y == 0.1
     assert msg.resolution.z == 0.1
 
+
 def test_number_of_rois(example_message):
     roi = PyRegionOfIntrest.from_message(example_message)
     assert roi.number_of_rois == 1
 
+
 def test_shape(example_message):
     roi = PyRegionOfIntrest.from_message(example_message)
     assert roi.shape == (399, 499, 599)
 
+
 def test_get_grid(example_message):
     roi = PyRegionOfIntrest.from_message(example_message)
     grid = roi.get_grid(0)
 
     assert grid.shape == (399, 499, 599, 3)
-    assert np.array_equal(grid[0, 0, 0], np.array([10.0 - 20.0, 20.0 - 25.0, 30.0 - 30.0]))
-    assert np.array_equal(grid[-1, -1, -1], np.array([10.0 + 20.0, 20.0 + 25.0, 30.0 + 30.0]))
+    assert np.array_equal(
+        grid[0, 0, 0], np.array([10.0 - 20.0, 20.0 - 25.0, 30.0 - 30.0])
+    )
+    assert np.array_equal(
+        grid[-1, -1, -1], np.array([10.0 + 20.0, 20.0 + 25.0, 30.0 + 30.0])
+    )
+
 
 def test_next_neighbor():
     grid_mm = np.zeros((20, 30, 40, 3))
diff --git a/test/test_volume.py b/test/test_volume.py
index e2a8ff8..a3c095b 100644
--- a/test/test_volume.py
+++ b/test/test_volume.py
@@ -1,6 +1,9 @@
 import pytest
 import numpy as np
-from rq_controller.common import PyVolume, PyRegionOfIntrest  # Adjust this import according to your module's path
+from rq_controller.common import (
+    PyVolume,
+    PyRegionOfIntrest,
+)  # Adjust this import according to your module's path
 from rq_controller.common.volume import VOLUME_TYPES
 from rq_interfaces.msg import Volume
 from visualization_msgs.msg import Marker
@@ -22,19 +25,19 @@ def example_message():
     marker.header.frame_id = "test_frame"
 
     msg.grid.region_of_intrest_stack.markers.append(marker)
-    
-    msg.grid.resolution.x = 1.
-    msg.grid.resolution.y = 1.
-    msg.grid.resolution.z = 1.
+
+    msg.grid.resolution.x = 1.0
+    msg.grid.resolution.y = 1.0
+    msg.grid.resolution.z = 1.0
 
     msg.datatype = VOLUME_TYPES.UINT_16
 
     shape = (40, 50, 60)
     slices = np.random.randint(0, 65535, size=shape, dtype=np.uint16)
     for i in range(shape[2]):
-        image_msg = ros2_numpy.msgify(Image, slices[:, :, i], encoding='mono16')
+        image_msg = ros2_numpy.msgify(Image, slices[:, :, i], encoding="mono16")
         msg.slices.append(image_msg)
-    
+
     return msg
 
 
@@ -73,7 +76,7 @@ def test_as_message(example_message):
     for i, slice_msg in enumerate(msg.slices):
         slice_array = ros2_numpy.numpify(slice_msg)
         assert np.array_equal(slice_array, volume.array[:, :, i])
- 
+
 
 def test_get_data_type():
     assert PyVolume.get_data_type(VOLUME_TYPES.UINT_16) == np.uint16
@@ -83,8 +86,8 @@ def test_get_data_type():
 
 
 def test_enum_to_numpify():
-    assert PyVolume.enum_to_numpify(VOLUME_TYPES.UINT_16) == 'mono16'
-    assert PyVolume.enum_to_numpify(VOLUME_TYPES.UINT_8) == 'mono8'
+    assert PyVolume.enum_to_numpify(VOLUME_TYPES.UINT_16) == "mono16"
+    assert PyVolume.enum_to_numpify(VOLUME_TYPES.UINT_8) == "mono8"
     with pytest.raises(ValueError):
         PyVolume.enum_to_numpify(-1)
 
@@ -95,4 +98,3 @@ def test_shape():
     volume = PyVolume(array, roi, VOLUME_TYPES.UINT_8)
 
     assert volume.shape == array.shape
-
-- 
GitLab