Source code for simtel.simulator_ray_tracing

"""Simulation runner for ray tracing simulations."""

import logging
from collections import namedtuple

import astropy.units as u

from simtools.io_operations import io_handler
from simtools.runners.simtel_runner import SimtelRunner
from simtools.utils import names

__all__ = ["SimulatorRayTracing"]

# pylint: disable=no-member
# The line above is needed because there are members which are created
# by adding them to the __dict__ of the class rather than directly.


[docs] class SimulatorRayTracing(SimtelRunner): """ Perform ray tracing simulations with sim_telarray. Parameters ---------- telescope_model: TelescopeModel telescope model label: str label used for output file naming. simtel_path: str or Path Location of sim_telarray installation. config_data: namedtuple namedtuple containing the configurable parameters as values (expected units in brackets): zenith_angle (deg), off_axis_angle (deg), source_distance (km), single_mirror_mode, use_random_focal_length, mirror_numbers. force_simulate: bool Remove existing files and force re-running of the ray-tracing simulation. """ def __init__( self, telescope_model, label=None, simtel_path=None, config_data=None, force_simulate=False, test=False, ): """Initialize SimtelRunner.""" self._logger = logging.getLogger(__name__) self._logger.debug("Init SimulatorRayTracing") super().__init__(label=label, simtel_path=simtel_path) self.telescope_model = telescope_model self.label = label if label is not None else self.telescope_model.label self.io_handler = io_handler.IOHandler() self._base_directory = self.io_handler.get_output_directory(self.label, "ray-tracing") self.config = ( self._config_to_namedtuple(config_data) if isinstance(config_data, dict) else config_data ) self._rep_number = 0 self.runs_per_set = 1 if self.config.single_mirror_mode else 20 self.photons_per_run = 100000 if not test else 5000 self._load_required_files(force_simulate) def _load_required_files(self, force_simulate): """ Load required files for the simulation. Depends on the running mode. Initialize files for the simulation. Parameters ---------- force_simulate: bool Remove existing files and force re-running of the ray-tracing simulation. """ # This file is not actually needed and does not exist in simtools. # It is required as CORSIKA input file to sim_telarray self._corsika_file = self._simtel_path.joinpath("run9991.corsika.gz") # Loop to define and remove existing files. # Files will be named _base_file = self.__dict__['_' + base + 'File'] for base_name in ["stars", "photons", "log"]: file_name = names.generate_file_name( file_type=base_name, suffix=".log" if base_name == "log" else ".lis", site=self.telescope_model.site, telescope_model_name=self.telescope_model.name, source_distance=( None if self.config.single_mirror_mode else self.config.source_distance ), zenith_angle=self.config.zenith_angle, off_axis_angle=self.config.off_axis_angle, mirror_number=( self.config.mirror_numbers if self.config.single_mirror_mode else None ), label=self.label, ) file = self._base_directory.joinpath(file_name) if file.exists() and force_simulate: file.unlink() # Defining the file name variable as an class attribute. self.__dict__["_" + base_name + "_file"] = file if not file.exists() or force_simulate: # Adding header to photon list file. with self._photons_file.open("w", encoding="utf-8") as file: file.write(f"#{50 * '='}\n") file.write("# List of photons for RayTracing simulations\n") file.write(f"#{50 * '='}\n") file.write(f"# config_file = {self.telescope_model.get_config_file()}\n") file.write(f"# zenith_angle [deg] = {self.config.zenith_angle}\n") file.write(f"# off_axis_angle [deg] = {self.config.off_axis_angle}\n") file.write(f"# source_distance [km] = {self.config.source_distance}\n") if self.config.single_mirror_mode: file.write(f"# mirror_number = {self.config.mirror_numbers}\n\n") # Filling a star file with a single light source defined by # - azimuth # - elevation # - flux # - distance of light source with self._stars_file.open("w", encoding="utf-8") as file: file.write( f"0. {90.0 - self.config.zenith_angle} 1.0 {self.config.source_distance}\n" ) if self.config.single_mirror_mode: self._logger.debug("For single mirror mode, need to prepare the single pixel camera.") self._write_out_single_pixel_camera_file() def _make_run_command( self, run_number=None, input_file=None ): # pylint: disable=unused-argument """Generate simtel_array run command.""" if self.config.single_mirror_mode: # Note: no mirror length defined for dual-mirror telescopes _mirror_focal_length = float( self.telescope_model.get_parameter_value("mirror_focal_length") ) # RayTracing command = str(self._simtel_path.joinpath("sim_telarray/bin/sim_telarray")) command += f" -c {self.telescope_model.get_config_file()}" command += f" -I{self.telescope_model.config_file_directory}" command += super().get_config_option("random_state", "none") command += super().get_config_option("IMAGING_LIST", str(self._photons_file)) command += super().get_config_option("stars", str(self._stars_file)) command += super().get_config_option( "altitude", self.telescope_model.get_parameter_value("corsika_observation_level") ) command += super().get_config_option( "telescope_theta", self.config.zenith_angle + self.config.off_axis_angle, ) command += super().get_config_option("star_photons", str(self.photons_per_run)) command += super().get_config_option("telescope_phi", "0") command += super().get_config_option("camera_transmission", "1.0") command += super().get_config_option("nightsky_background", "all:0.") command += super().get_config_option("trigger_current_limit", "1e10") command += super().get_config_option("telescope_random_angle", "0") command += super().get_config_option("telescope_random_error", "0") command += super().get_config_option("convergent_depth", "0") command += super().get_config_option("maximum_telescopes", "1") command += super().get_config_option("show", "all") command += super().get_config_option("camera_filter", "none") if self.config.single_mirror_mode: command += super().get_config_option("focus_offset", "all:0.") command += super().get_config_option("camera_config_file", "single_pixel_camera.dat") command += super().get_config_option("camera_pixels", "1") command += super().get_config_option("trigger_pixels", "1") command += super().get_config_option("camera_body_diameter", "0") command += super().get_config_option( "mirror_list", self.telescope_model.get_single_mirror_list_file( self.config.mirror_numbers, self.config.use_random_focal_length ), ) command += super().get_config_option( "focal_length", self.config.source_distance * u.km.to(u.cm) ) command += super().get_config_option("dish_shape_length", _mirror_focal_length) command += super().get_config_option("mirror_focal_length", _mirror_focal_length) command += super().get_config_option("parabolic_dish", "0") command += super().get_config_option("mirror_align_random_distance", "0.") command += super().get_config_option("mirror_align_random_vertical", "0.,28.,0.,0.") command += " " + str(self._corsika_file) return command, self._log_file, self._log_file def _check_run_result(self, run_number=None): # pylint: disable=unused-argument """ Check run results. Photon list files should have at least 100 lines. Returns ------- bool True if photon list is not empty. Raises ------ RuntimeError if Photon list is empty. """ with open(self._photons_file, "rb") as ff: n_lines = sum(1 for _ in ff) if n_lines < 100: raise RuntimeError("Photon list is empty.") return True def _write_out_single_pixel_camera_file(self): """Write out the single pixel camera file.""" with self.telescope_model.config_file_directory.joinpath("single_pixel_camera.dat").open( "w" ) as file: file.write("# Single pixel camera\n") file.write('PixType 1 0 0 300 1 300 0.00 "funnel_perfect.dat"\n') file.write("Pixel 0 1 0. 0. 0 0 0 0x00 1\n") file.write("Trigger 1 of 0\n") # need to also write out the funnel_perfect.dat file with self.telescope_model.config_file_directory.joinpath("funnel_perfect.dat").open( "w" ) as file: file.write( "# Perfect light collection where the angular efficiency of funnels is needed\n" ) file.write("0 1.0\n") file.write("30 1.0\n") file.write("60 1.0\n") file.write("90 1.0\n") def _config_to_namedtuple(self, data_dict): """Convert dict to namedtuple for configuration.""" config_data = namedtuple( "Config", [ "zenith_angle", "off_axis_angle", "source_distance", "single_mirror_mode", "use_random_focal_length", "mirror_numbers", ], ) return config_data( zenith_angle=data_dict["zenith_angle"], off_axis_angle=data_dict["off_axis_angle"], source_distance=data_dict["source_distance"], single_mirror_mode=data_dict["single_mirror_mode"], use_random_focal_length=data_dict["use_random_focal_length"], mirror_numbers=data_dict["mirror_numbers"], )