Source code for model.mirrors

"""Definition and modeling of mirror panels."""

import logging
from pathlib import Path

import astropy.io.ascii
import astropy.units as u
import numpy as np
from astropy.table import Table

__all__ = ["InvalidMirrorListFileError", "Mirrors"]


[docs] class InvalidMirrorListFileError(Exception): """Exception for invalid mirror list file."""
[docs] class Mirrors: """ Mirrors class, created from a mirror list file. Parameters ---------- mirror_list_file: Union[str, Path] Mirror list in sim_telarray or ecsv format (with panel focal length only). parameters: dict, optional Dictionary of parameters from the database. """ def __init__(self, mirror_list_file: str | Path, parameters: dict | None = None): """Initialize Mirrors.""" self._logger = logging.getLogger(__name__) self._logger.debug("Mirrors Init") self.mirror_table = Table() self.mirror_diameter = None self.shape_type = None self.number_of_mirrors = 0 self.parameters = parameters self._mirror_list_file = mirror_list_file self._read_mirror_list() def _read_mirror_list(self): """ Read the mirror lists from disk and store the data. Allow reading of mirror lists in sim_telarray and ecsv format. """ if str(self._mirror_list_file).find("ecsv") > 0: self._read_mirror_list_from_ecsv() else: self._read_mirror_list_from_sim_telarray() def _read_mirror_list_from_ecsv(self): """ Read the mirror list in ecsv format and store the data. Raises ------ InvalidMirrorListFileError If number of mirrors is 0. """ self._logger.debug(f"Reading mirror properties from {self._mirror_list_file}") self.mirror_table = Table.read(self._mirror_list_file, format="ascii.ecsv") self.number_of_mirrors = np.shape(self.mirror_table)[0] self._logger.debug(f"Number of Mirrors = {self.number_of_mirrors}") if self.number_of_mirrors == 0: msg = "Problem reading mirror list file" self._logger.error(msg) raise InvalidMirrorListFileError try: self.mirror_diameter = u.Quantity(self.mirror_table["mirror_diameter"])[0] self._logger.debug(f"Mirror diameter = {self.mirror_diameter}") except KeyError: self._logger.debug("Mirror mirror_panel_diameter not in mirror file") try: self.mirror_diameter = u.Quantity( self.parameters["mirror_panel_diameter"]["value"], self.parameters["mirror_panel_diameter"]["unit"], ) self._logger.debug("Take mirror_panel_diameter from parameters") except TypeError as error: msg = "Mirror mirror_panel_diameter not contained in DB" self._logger.error(msg) raise TypeError(msg) from error if "focal_length" not in self.mirror_table.colnames: try: self.mirror_table["focal_length"] = ( self.mirror_table["mirror_curvature_radius"].to("cm") / 2 ) except KeyError: self._logger.debug("mirror_curvature_radius not contained in mirror list") try: self.mirror_table["focal_length"] = self.number_of_mirrors * [ u.Quantity( self.parameters["mirror_focal_length"]["value"], self.parameters["mirror_focal_length"]["unit"], ) ] self._logger.debug("Take mirror_focal_length from parameters") except TypeError as error: msg = "mirror_focal_length not contained in DB" self._logger.error(msg) raise TypeError(msg) from error try: self.shape_type = u.Quantity(self.mirror_table["shape_type"])[0] self._logger.debug(f"Mirror shape_type = {self.shape_type}") except KeyError: self._logger.debug("Mirror shape_type not in mirror file") try: self.shape_type = self.parameters["mirror_panel_shape"]["value"] self._logger.debug("Take shape_type from parameters") except TypeError as error: msg = "Mirror shape_type not contained in DB" self._logger.error(msg) raise TypeError(msg) from error def _read_mirror_list_from_sim_telarray(self): """ Read the mirror list in sim_telarray format and store the data. Allow to read mirror lists with different number of columns. Raises ------ InvalidMirrorListFileError If number of mirrors is 0. """ self._logger.debug(f"Reading mirror properties from {self._mirror_list_file}") try: self.mirror_table = Table.read( self._mirror_list_file, format="ascii.no_header", names=[ "mirror_x", "mirror_y", "mirror_diameter", "focal_length", "shape_type", "mirror_z", "sep", "mirror_panel_id", ], units=["cm", "cm", "cm", "cm", None, "cm", None, None], ) self.mirror_table["mirror_panel_id"] = np.array( [ int("".join(filter(str.isdigit, string))) for string in self.mirror_table["mirror_panel_id"] ] ) except astropy.io.ascii.core.InconsistentTableError: self._logger.debug("Try and read mirror list with low number of columns") self.mirror_table = Table.read( self._mirror_list_file, format="ascii.no_header", names=[ "mirror_x", "mirror_y", "mirror_diameter", "focal_length", "shape_type", ], units=["cm", "cm", "cm", "cm", None], ) self.mirror_table["mirror_panel_id"] = np.arange(len(self.mirror_table["mirror_x"])) self.shape_type = self.mirror_table["shape_type"][0] self.mirror_diameter = u.Quantity( self.mirror_table["mirror_diameter"][0], self.mirror_table["mirror_diameter"].unit ) self.number_of_mirrors = len(self.mirror_table["focal_length"]) self._logger.debug(f"Mirror shape_type = {self.shape_type}") self._logger.debug(f"Mirror diameter = {self.mirror_diameter}") self._logger.debug(f"Number of Mirrors = {self.number_of_mirrors}")
[docs] def get_single_mirror_parameters(self, number: int) -> tuple: """ Get parameters for a single mirror given by number. Parameters ---------- number: int Mirror number of desired parameters. Returns ------- tuple (pos_x, pos_y, mirror_diameter, focal_length, shape_type): tuple of float X, Y positions, mirror_diameter, focal length and shape_type. """ mask = self.mirror_table["mirror_panel_id"] == number if not np.any(mask): self._logger.debug(f"Mirror id{number} not in table, using first mirror instead") mask[0] = True try: return_values = ( u.Quantity( self.mirror_table[mask]["mirror_x"].value[0], self.mirror_table[mask]["mirror_x"].unit, ), u.Quantity( self.mirror_table[mask]["mirror_y"].value[0], self.mirror_table[mask]["mirror_y"].unit, ), u.Quantity( self.mirror_table[mask]["mirror_diameter"].value[0], self.mirror_table[mask]["mirror_diameter"].unit, ), u.Quantity( self.mirror_table[mask]["focal_length"].value[0], self.mirror_table[mask]["focal_length"].unit, ), u.Quantity( self.mirror_table[mask]["shape_type"].value[0], self.mirror_table[mask]["shape_type"].unit, ), ) except KeyError: self._logger.debug("Mirror list missing required column") return_values = ( 0, 0, self.mirror_diameter, u.Quantity( self.mirror_table[mask]["focal_length"].value[0], self.mirror_table[mask]["focal_length"].unit, ), self.shape_type, ) return return_values
[docs] def plot_mirror_layout(self): """Plot the mirror layout (not implemented yet)."""