Source code for nxtomomill.converter.hdf5.acquisition.blisstomoconfig

from __future__ import annotations

import numpy
import h5py
from silx.io.utils import h5py_read_dataset
from nxtomomill.models.utils import convert_str_to_tuple, convert_str_to_bool

__all__ = [
    "TomoConfig",
]


[docs]class TomoConfig: """ hold motor used for tomography acquisition according to https://tomo.gitlab-pages.esrf.fr/bliss-tomo/master/modelization_sample_stage.html convension """
[docs] def __init__(self) -> None: self._rotation: None | tuple[str, ...] = None self._sample_u: None | tuple[str, ...] = None self._sample_v: None | tuple[str, ...] = None self._sample_x: None | tuple[str, ...] = None self._sample_y: None | tuple[str, ...] = None self._translation_x: None | tuple[str, ...] = None self._translation_y: None | tuple[str, ...] = None self._translation_z: None | tuple[str, ...] = None self._tomo_detector: None | tuple[str, ...] = None self._rotation_is_clockwise: None | bool = None
@property def rotation(self) -> tuple[str, ...] | None: return self._rotation @rotation.setter def rotation(self, motor: tuple[str, ...] | None): self._rotation = motor @property def sample_u(self) -> tuple[str, ...] | None: return self._sample_u @sample_u.setter def sample_u(self, motor: tuple[str, ...] | None): self._sample_u = motor @property def sample_v(self) -> tuple[str, ...] | None: return self._sample_v @sample_v.setter def sample_v(self, motor: tuple[str, ...] | None): self._sample_v = motor @property def sample_x(self) -> tuple[str, ...] | None: return self._sample_x @sample_x.setter def sample_x(self, motor: tuple[str, ...] | None): self._sample_x = motor @property def sample_y(self) -> tuple[str, ...] | None: return self._sample_y @sample_y.setter def sample_y(self, motor: tuple[str, ...] | None): self._sample_y = motor @property def translation_x(self) -> tuple[str, ...] | None: return self._translation_x @translation_x.setter def translation_x(self, motor: tuple[str, ...] | None): self._translation_x = motor @property def translation_y(self) -> tuple[str, ...] | None: return self._translation_y @translation_y.setter def translation_y(self, motor: tuple[str, ...] | None): self._translation_y = motor @property def translation_z(self) -> tuple[str, ...] | None: return self._translation_z @translation_z.setter def translation_z(self, motor: tuple[str, ...] | None): self._translation_z = motor @property def tomo_detector(self) -> tuple[str, ...] | None: return self._tomo_detector @tomo_detector.setter def tomo_detector(self, detector_name: tuple[str, ...]): self._tomo_detector = detector_name @property def rotation_is_clockwise(self) -> bool | None: return self._rotation_is_clockwise @rotation_is_clockwise.setter def rotation_is_clockwise(self, is_clockwise: bool | None) -> None: if is_clockwise is None: self._rotation_is_clockwise = is_clockwise elif isinstance(is_clockwise, (bool, numpy.bool_)): self._rotation_is_clockwise = bool(is_clockwise) else: raise TypeError( f"'is_clockwise' should be a bool or Not. Got {type(is_clockwise)}" ) def __str__(self) -> str: return "tomo_config:" + " ; ".join( [ f"rotation={', '.join(self.rotation)}", f"rotation_is_clockwise={self.rotation_is_clockwise}", f"sample_u={', '.join(self.sample_u)}", f"sample_v={', '.join(self.sample_v)}", f"sample_x={', '.join(self.sample_x)}", f"sample_y={', '.join(self.sample_y)}", f"tomo_detector={', '.join(self.tomo_detector)}", f"translation_x={', '.join(self.translation_x)}", f"translation_y={', '.join(self.translation_y)}", f"translation_z={', '.join(self.translation_z)}", ] )
[docs] @staticmethod def from_technique_group(technique_group: h5py.Group): """ get rotation motor and thinks like this from the 'tomoconfig'. This can retrieve one or several dataset name or a single one. In the case of several dataset name we get (real_motor_name, bliss_alias) If the motor moves then this is pretty simple the real motor_name dataset exists. But if the motors does not move during the bliss scan (scalar value) then the real_motor_name dataset doesn't exists and the bliss alias does. This is why we need to keep both and check both during the 'standard process'... """ if not isinstance(technique_group, h5py.Group): raise TypeError( f"instrument_group is expected to be an instance of {h5py.Group}. {type(technique_group)} provided" ) if "tomoconfig" not in technique_group: raise KeyError("could find 'tomoconfig' key") else: tomo_config_group = technique_group.get("tomoconfig") def get_dataset(group, dataset_name, default, dtype=tuple): if dataset_name not in group: return default dataset = h5py_read_dataset(group[dataset_name]) # note: the dataset are read as string '["detector"]' and must be converted to dtype if dtype is tuple: return convert_str_to_tuple(tuple(dataset)) elif dtype is bool: return convert_str_to_bool(dataset) else: raise TypeError(f"invalid dtype: {dtype}") tomo_config = TomoConfig() tomo_config.rotation = get_dataset(tomo_config_group, "rotation", None) tomo_config.rotation_is_clockwise = get_dataset( tomo_config_group, "rotation_is_clockwise", default=None, dtype=bool ) tomo_config.sample_u = get_dataset(tomo_config_group, "sample_u", None) tomo_config.sample_v = get_dataset(tomo_config_group, "sample_v", None) tomo_config.sample_x = get_dataset(tomo_config_group, "sample_x", None) tomo_config.sample_y = get_dataset(tomo_config_group, "sample_y", None) tomo_config.tomo_detector = get_dataset(tomo_config_group, "detector", None) # translation x == sample_u tomo_config.translation_x = get_dataset( tomo_config_group, "sample_u", None ) # Needs translation X mapping tomo_config.translation_y = get_dataset( tomo_config_group, "translation_y", None ) tomo_config.translation_z = get_dataset( tomo_config_group, "translation_z", None ) return tomo_config