"""MC model of a telescope."""
import logging
from pathlib import Path
import astropy.io.ascii
import numpy as np
from astropy.table import Table
import simtools.utils.general as gen
from simtools.model.camera import Camera
from simtools.model.mirrors import Mirrors
from simtools.model.model_parameter import InvalidModelParameterError, ModelParameter
from simtools.utils import names
__all__ = ["TelescopeModel"]
[docs]
class TelescopeModel(ModelParameter):
"""
TelescopeModel represents the MC model of an individual telescope.
TelescopeModel contains parameter names and values for a specific telescope model.
Parameters
----------
site: str
Site name (e.g., South or North).
telescope_name: str
Telescope name (ex. LSTN-01, LSTN-design, ...).
mongo_db_config: dict
MongoDB configuration.
model_version: str
Model version.
label: str, optional
Instance label. Important for output file naming.
"""
def __init__(
self,
site: str,
telescope_name: str,
mongo_db_config: dict,
model_version: str,
label: str | None = None,
):
"""Initialize TelescopeModel."""
super().__init__(
site=site,
array_element_name=telescope_name,
mongo_db_config=mongo_db_config,
model_version=model_version,
db=None,
label=label,
)
self._logger = logging.getLogger(__name__)
self._logger.debug("Init TelescopeModel %s %s", site, telescope_name)
self._single_mirror_list_file_paths = None
self._mirrors = None
self._camera = None
@property
def mirrors(self):
"""Load the mirror information if the class instance hasn't done it yet."""
if self._mirrors is None:
self._load_mirrors()
return self._mirrors
@property
def camera(self):
"""Load the camera information if the class instance hasn't done it yet."""
if self._camera is None:
self._load_camera()
return self._camera
[docs]
def export_single_mirror_list_file(self, mirror_number: int, set_focal_length_to_zero: bool):
"""
Export a mirror list file with a single mirror in it.
Parameters
----------
mirror_number: int
Number index of the mirror.
set_focal_length_to_zero: bool
Set the focal length to zero if True.
"""
if mirror_number > self.mirrors.number_of_mirrors:
logging.error("mirror_number > number_of_mirrors")
return
file_name = names.simtel_single_mirror_list_file_name(
self.site, self.name, self.model_version, mirror_number, self.label
)
if self._single_mirror_list_file_paths is None:
self._single_mirror_list_file_paths = {}
self._single_mirror_list_file_paths[mirror_number] = self.config_file_directory.joinpath(
file_name
)
# Using SimtelConfigWriter
self._load_simtel_config_writer()
self.simtel_config_writer.write_single_mirror_list_file(
mirror_number,
self.mirrors,
self._single_mirror_list_file_paths[mirror_number],
set_focal_length_to_zero,
)
[docs]
def get_single_mirror_list_file(
self, mirror_number: int, set_focal_length_to_zero: bool = False
):
"""
Get the path to the single mirror list file.
Parameters
----------
mirror_number: int
Mirror number.
set_focal_length_to_zero: bool
Flag to set the focal length to zero.
Returns
-------
Path
Path of the single mirror list file.
"""
self.export_single_mirror_list_file(mirror_number, set_focal_length_to_zero)
return self._single_mirror_list_file_paths[mirror_number]
def _load_mirrors(self):
"""Load the attribute mirrors by creating a Mirrors object with the mirror list file."""
mirror_list_file_name = self.get_parameter_value("mirror_list")
self._logger.debug(f"Reading mirror list from {mirror_list_file_name}")
try:
mirror_list_file = gen.find_file(mirror_list_file_name, self.config_file_directory)
except FileNotFoundError:
mirror_list_file = gen.find_file(mirror_list_file_name, self.io_handler.model_path)
self._logger.warning(
"Mirror_list_file was not found in the config directory - "
"Using the one found in the model_path"
)
except TypeError as exc:
raise TypeError("Undefined mirror list") from exc
self._mirrors = Mirrors(mirror_list_file, parameters=self._parameters)
[docs]
def get_telescope_effective_focal_length(
self, unit: str = "m", return_focal_length_if_zero: bool = False
) -> float:
"""
Return effective focal length.
The function ensures backwards compatibility with older sim-telarray versions.
Parameters
----------
unit: str
Unit of the effective focal length. Default is 'm'.
return_focal_length_if_zero: bool
If True, return the focal length if the effective focal length is 0.
Returns
-------
float:
Effective focal length.
"""
try:
eff_focal_length = self.get_parameter_value_with_unit("effective_focal_length")[0]
except TypeError:
eff_focal_length = self.get_parameter_value_with_unit("effective_focal_length")
try:
eff_focal_length = eff_focal_length.to(unit).value
except AttributeError:
eff_focal_length = 0.0
if return_focal_length_if_zero and (eff_focal_length is None or eff_focal_length == 0.0):
self._logger.warning("Using focal_length because effective_focal_length is 0")
return self.get_parameter_value_with_unit("focal_length").to(unit).value
return eff_focal_length
def _load_camera(self):
"""Load camera attribute by creating a Camera object with the camera config file."""
camera_config_file = self.get_parameter_value("camera_config_file")
focal_length = self.get_telescope_effective_focal_length("cm", True)
try:
camera_config_file_path = gen.find_file(camera_config_file, self.config_file_directory)
except TypeError as exc:
self._logger.error(
f"Camera config file {camera_config_file} or "
f"config file directory ({self.config_file_directory}) is None"
)
raise TypeError from exc
except FileNotFoundError:
self._logger.warning(
f"Camera config file {camera_config_file} not found in the config directory "
f"{self.config_file_directory}. Using the one found in the model_path"
)
camera_config_file_path = gen.find_file(camera_config_file, self.io_handler.model_path)
self._camera = Camera(
telescope_model_name=self.name,
camera_config_file=camera_config_file_path,
focal_length=focal_length,
)
[docs]
def is_file_2d(self, par: str) -> bool:
"""
Check if the file referenced by par is a 2D table.
Parameters
----------
par: str
Name of the parameter.
Returns
-------
bool:
True if the file is a 2D map type.
"""
try:
file_name = self.get_parameter_value(par)
except KeyError:
logging.error(f"Parameter {par} does not exist")
return False
file = self.config_file_directory.joinpath(file_name)
with open(file, encoding="utf-8") as f:
return "@RPOL@" in f.read()
[docs]
def read_two_dim_wavelength_angle(self, file_name: str | Path) -> dict:
"""
Read a two dimensional distribution of wavelength and angle (z-axis can be anything).
Return a dictionary with three arrays, wavelength, angles, z (can be transmission,
reflectivity, etc.)
Parameters
----------
file_name: str or Path
File assumed to be in the model directory.
Returns
-------
dict:
dict of three arrays, wavelength, degrees, z.
"""
_file = self.config_file_directory.joinpath(file_name)
self._logger.debug("Reading two dimensional distribution from %s", _file)
line_to_start_from = 0
with open(_file, encoding="utf-8") as f:
for i_line, line in enumerate(f):
if line.startswith("ANGLE"):
degrees = np.array(line.strip().split("=")[1].split(), dtype=np.float16)
line_to_start_from = i_line + 1
break # The rest can be read with np.loadtxt
_data = np.loadtxt(_file, skiprows=line_to_start_from)
return {
"Wavelength": _data[:, 0],
"Angle": degrees,
"z": np.array(_data[:, 1:]).T,
}
[docs]
def get_on_axis_eff_optical_area(self) -> float:
"""Return the on-axis effective optical area (derived previously for this telescope)."""
ray_tracing_data = astropy.io.ascii.read(
self.config_file_directory.joinpath(self.get_parameter_value("optics_properties"))
)
if not np.isclose(ray_tracing_data["Off-axis angle"][0], 0):
self._logger.error(
f"No value for the on-axis effective optical area exists."
f" The minimum off-axis angle is {ray_tracing_data['Off-axis angle'][0]}"
)
raise ValueError
return ray_tracing_data["eff_area"][0]
[docs]
def read_incidence_angle_distribution(self, incidence_angle_dist_file: str) -> Table:
"""
Read the incidence angle distribution from a file.
Parameters
----------
incidence_angle_dist_file: str
File name of the incidence angle distribution
Returns
-------
incidence_angle_dist: astropy.table.Table
Instance of astropy.table.Table with the incidence angle distribution.
"""
self._logger.debug(
"Reading incidence angle distribution from %s",
self.config_file_directory.joinpath(incidence_angle_dist_file),
)
return astropy.io.ascii.read(self.config_file_directory.joinpath(incidence_angle_dist_file))
[docs]
@staticmethod
def calc_average_curve(curves: dict, incidence_angle_dist: Table) -> Table:
"""
Calculate an average curve from a set of curves.
The calculation uses weights the distribution of incidence angles provided in
incidence_angle_dist.
Parameters
----------
curves: dict
dict of with 3 "columns", Wavelength, Angle and z. The dictionary represents a two \
dimensional distribution of wavelengths and angles with the z value being e.g., \
reflectivity, transmission, etc.
incidence_angle_dist: astropy.table.Table
Instance of astropy.table.Table with the incidence angle distribution. The assumed \
columns are "Incidence angle" and "Fraction".
Returns
-------
average_curve: astropy.table.Table
Instance of astropy.table.Table with the averaged curve.
"""
weights = []
for angle_now in curves["Angle"]:
weights.append(
incidence_angle_dist["Fraction"][
np.nanargmin(np.abs(angle_now - incidence_angle_dist["Incidence angle"].value))
]
)
return Table(
[curves["Wavelength"], np.average(curves["z"], weights=weights, axis=0)],
names=("Wavelength", "z"),
)
[docs]
def export_table_to_model_directory(self, file_name: str, table: Table) -> str:
"""
Write out a file with the provided table to the model directory.
Parameters
----------
file_name: str
File name to write to.
table: astropy.table.Table
Instance of astropy.table.Table with the values to write to the file.
Returns
-------
Path:
Path to the file exported.
"""
file_to_write_to = self.config_file_directory.joinpath(file_name)
table.write(file_to_write_to, format="ascii.commented_header", overwrite=True)
return file_to_write_to.absolute()
[docs]
def position(self, coordinate_system: str = "ground") -> list:
"""
Get coordinates in the given system.
Parameters
----------
coordinate_system: str
Coordinates system. Default is 'ground'.
Returns
-------
list :
List of telescope position in the requested coordinate system.
Raises
------
KeyError
If the coordinate system is not found.
"""
try:
return self.get_parameter_value_with_unit(f"array_element_position_{coordinate_system}")
except InvalidModelParameterError as exc:
self._logger.error(f"Coordinate system {coordinate_system} not found.")
raise exc