"""Base class for running sim_telarray simulations."""
import logging
import stat
import subprocess
from pathlib import Path
import simtools.utils.general as gen
from simtools.runners.runner_services import RunnerServices
__all__ = ["InvalidOutputFileError", "SimtelExecutionError", "SimtelRunner"]
[docs]
class SimtelExecutionError(Exception):
"""Exception for simtel_array execution error."""
[docs]
class InvalidOutputFileError(Exception):
"""Exception for invalid output file."""
[docs]
class SimtelRunner:
"""
Base class for running sim_telarray simulations.
Parameters
----------
simtel_path: str or Path
Location of sim_telarray installation.
label: str
Instance label. Important for output file naming.
"""
def __init__(self, simtel_path, label=None, corsika_config=None, use_multipipe=False):
"""Initialize SimtelRunner."""
self._logger = logging.getLogger(__name__)
self._simtel_path = Path(simtel_path)
self.label = label
self._base_directory = None
self.runs_per_set = 1
self.runner_service = RunnerServices(corsika_config, label)
self._directory = self.runner_service.load_data_directories(
"corsika_simtel" if use_multipipe else "simtel"
)
def __repr__(self):
"""Return a string representation of the SimtelRunner object."""
return f"SimtelRunner(label={self.label})\n"
[docs]
def prepare_run_script(self, test=False, input_file=None, run_number=None, extra_commands=None):
"""
Build and return the full path of the bash run script containing the sim_telarray command.
Parameters
----------
test: bool
Test flag for faster execution.
input_file: str or Path
Full path of the input CORSIKA file.
run_number: int
Run number.
extra_commands: str
Additional commands for running simulations given in config.yml.
Returns
-------
Path
Full path of the run script.
"""
self._logger.debug("Creating run bash script")
script_file_path = self.get_file_name(file_type="sub_script", run_number=run_number)
self._logger.debug(f"Run bash script - {script_file_path}")
self._logger.debug(f"Extra commands to be added to the run script {extra_commands}")
command = self._make_run_command(run_number=run_number, input_file=input_file)
with script_file_path.open("w", encoding="utf-8") as file:
file.write("#!/usr/bin/env bash\n\n")
file.write("set -e\n")
file.write("set -o pipefail\n")
file.write("\nSECONDS=0\n")
if extra_commands is not None:
file.write("# Writing extras\n")
for line in extra_commands:
file.write(f"{line}\n")
file.write("# End of extras\n\n")
n = 1 if test else self.runs_per_set
for _ in range(n):
file.write(f"{command}\n\n")
file.write('\necho "RUNTIME: $SECONDS"\n')
script_file_path.chmod(script_file_path.stat().st_mode | stat.S_IXUSR | stat.S_IXGRP)
return script_file_path
[docs]
def run(self, test=False, input_file=None, run_number=None):
"""
Make run command and run sim_telarray.
Parameters
----------
test: bool
If True, make simulations faster.
input_file: str or Path
Full path of the input CORSIKA file.
run_number: int
Run number.
"""
self._logger.debug("Running sim_telarray")
command, stdout_file, stderr_file = self._make_run_command(
run_number=run_number, input_file=input_file
)
if test:
self._logger.info(f"Running (test) with command: {command}")
self._run_simtel_and_check_output(command, stdout_file, stderr_file)
else:
self._logger.debug(f"Running ({self.runs_per_set}x) with command: {command}")
for _ in range(self.runs_per_set):
self._run_simtel_and_check_output(command, stdout_file, stderr_file)
self._check_run_result(run_number=run_number)
def _check_run_result(self, run_number=None): # pylint: disable=all
"""Check if simtel output file exists."""
pass
def _raise_simtel_error(self):
"""
Raise sim_telarray execution error.
Final 30 lines from the log file are collected and printed.
Raises
------
SimtelExecutionError
"""
if hasattr(self, "_log_file"):
msg = gen.get_log_excerpt(self._log_file)
else:
msg = "Simtel log file does not exist."
self._logger.error(msg)
raise SimtelExecutionError(msg)
def _run_simtel_and_check_output(self, command, stdout_file, stderr_file):
"""
Run the sim_telarray command and check the exit code.
Raises
------
SimtelExecutionError
if run was not successful.
"""
stdout_file = stdout_file if stdout_file else "/dev/null"
stderr_file = stderr_file if stderr_file else "/dev/null"
with (
open(f"{stdout_file}", "w", encoding="utf-8") as stdout,
open(f"{stderr_file}", "w", encoding="utf-8") as stderr,
):
result = subprocess.run(
command,
shell=True,
text=True,
stdout=stdout,
stderr=stderr,
)
if result.returncode != 0:
self._logger.error(result.stderr)
self._raise_simtel_error()
return result.returncode
def _make_run_command(self, run_number=None, input_file=None):
self._logger.debug(
"make_run_command is being called from the base class - "
"it should be implemented in the sub class"
)
input_file = input_file if input_file else "nofile"
run_number = run_number if run_number else 1
return f"{input_file}-{run_number}", None, None
[docs]
@staticmethod
def get_config_option(par, value=None, weak_option=False):
"""
Build sim_telarray command.
Parameters
----------
par: str
Parameter name.
value: str
Parameter value.
weak_option: bool
If True, use -W option instead of -C.
Returns
-------
str
Command for sim_telarray.
"""
option_syntax = "-W" if weak_option else "-C"
c = f" {option_syntax} {par}"
c += f"={value}" if value is not None else ""
return c
[docs]
def get_resources(self, run_number=None):
"""Return computing resources used."""
return self.runner_service.get_resources(run_number)
[docs]
def get_file_name(self, simulation_software="simtel", file_type=None, run_number=None, mode=""):
"""
Get the full path of a file for a given run number.
Parameters
----------
simulation_software: str
Simulation software.
file_type: str
File type.
run_number: int
Run number.
Returns
-------
str
File name with full path.
"""
if simulation_software.lower() != "simtel":
raise ValueError(
f"simulation_software ({simulation_software}) is not supported in SimulatorArray"
)
return self.runner_service.get_file_name(
file_type=file_type, run_number=run_number, mode=mode
)