Source code for runners.corsika_runner

"""Generate run scripts and directories for CORSIKA simulations."""

import logging
import stat
from pathlib import Path

from simtools.io_operations import io_handler
from simtools.runners.runner_services import RunnerServices

__all__ = ["CorsikaRunner", "MissingRequiredEntryInCorsikaConfigError"]


[docs] class MissingRequiredEntryInCorsikaConfigError(Exception): """Exception for missing required entry in corsika config."""
[docs] class CorsikaRunner: """ Generate run scripts and directories for CORSIKA simulations. Run simulations if requested. CorsikaRunner is responsible for configuring and running CORSIKA, using corsika_autoinputs provided by the sim_telarray package. CorsikaRunner generates shell scripts to be run externally or by the simulator module simulator. CorsikaRunner is configured through a CorsikaConfig instance. Parameters ---------- corsika_config_data: CorsikaConfig CORSIKA configuration. simtel_path: str or Path Location of source of the sim_telarray/CORSIKA package. label: str Instance label. keep_seeds: bool Use seeds based on run number and primary particle. If False, use sim_telarray seeds. use_multipipe: bool Use multipipe to run CORSIKA and sim_telarray. """ def __init__( self, corsika_config, simtel_path, label=None, keep_seeds=False, use_multipipe=False, ): """Initialize CorsikaRunner.""" self._logger = logging.getLogger(__name__) self._logger.debug("Init CorsikaRunner") self.label = label self.corsika_config = corsika_config self._keep_seeds = keep_seeds self._use_multipipe = use_multipipe self._simtel_path = Path(simtel_path) self.io_handler = io_handler.IOHandler() self.runner_service = RunnerServices(corsika_config, label) self._directory = self.runner_service.load_data_directories("corsika")
[docs] def prepare_run_script( self, run_number=None, extra_commands=None, input_file=None, use_pfp=True ): """ Get the full path of the run script file for a given run number. Parameters ---------- use_pfp: bool Whether to use the preprocessor in preparing the CORSIKA input file run_number: int Run number. extra_commands: str Additional commands for running simulations. Returns ------- Path: Full path of the run script file. """ if input_file is not None: self._logger.warning( "input_file parameter is not used in CorsikaRunner.prepare_run_script" ) self.corsika_config.run_number = run_number script_file_path = self.get_file_name( file_type="sub_script", run_number=self.corsika_config.run_number ) corsika_input_file = self.corsika_config.generate_corsika_input_file( use_multipipe=self._use_multipipe, use_test_seeds=self._keep_seeds ) # CORSIKA input file for a specific run, created by the preprocessor pfp corsika_input_tmp_name = self.corsika_config.get_corsika_config_file_name( file_type="config_tmp", run_number=self.corsika_config.run_number ) corsika_input_tmp_file = self._directory["inputs"].joinpath(corsika_input_tmp_name) # CORSIKA log file naming (temporary and final) corsika_log_tmp_file = ( self._directory["data"] .joinpath(f"run{self.corsika_config.run_number:06}") .joinpath(f"run{self.corsika_config.run_number}.log") ) corsika_log_file = self.get_file_name( file_type="corsika_log", run_number=self.corsika_config.run_number ) if use_pfp: pfp_command = self._get_pfp_command(corsika_input_tmp_file, corsika_input_file) autoinputs_command = self._get_autoinputs_command( self.corsika_config.run_number, corsika_input_tmp_file ) self._logger.debug(f"Extra commands to be added to the run script: {extra_commands}") self._logger.debug(f"CORSIKA data will be set to {self._directory['data']}") with open(script_file_path, "w", encoding="utf-8") as file: file.write("#!/usr/bin/env bash\n") file.write("set -e\n") file.write("set -o pipefail\n") # Setting SECONDS variable to measure runtime file.write("\nSECONDS=0\n") if extra_commands is not None: file.write("\n# Writing extras\n") file.write(f"{extra_commands}\n") file.write("# End of extras\n\n") file.write(f"export CORSIKA_DATA={self._directory['data']}\n") file.write('mkdir -p "$CORSIKA_DATA"\n') file.write('cd "$CORSIKA_DATA" || exit 2\n') if use_pfp: file.write("\n# Running pfp\n") file.write(pfp_command) file.write("\n# Replacing the XXXXXX placeholder with the run number\n") file.write( f"sed -i 's/XXXXXX/{self.corsika_config.run_number:06}/g' " f"{corsika_input_tmp_file}\n" ) else: file.write("\n# Copying CORSIKA input file to run location\n") file.write(f"cp {corsika_input_file} {corsika_input_tmp_file}") file.write("\n# Running corsika_autoinputs\n") file.write(autoinputs_command) file.write("\n# Moving log files to the corsika log directory\n") file.write(f"gzip {corsika_log_tmp_file}\n") file.write(f"mv -v {corsika_log_tmp_file}.gz {corsika_log_file}\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 get_resources(self, run_number=None): """Return computing resources used.""" return self.runner_service.get_resources(run_number)
def _get_pfp_command(self, input_tmp_file, corsika_input_file): """ Get pfp pre-processor command. pfp is a pre-processor tool and part of sim_telarray. Parameters ---------- input_tmp_file: Path Temporary input file. Returns ------- str pfp command. """ cmd = self._simtel_path.joinpath("sim_telarray/bin/pfp") cmd = str(cmd) + f" -V -DWITHOUT_MULTIPIPE - < {corsika_input_file}" cmd += f" > {input_tmp_file} || exit\n" return cmd def _get_autoinputs_command(self, run_number, input_tmp_file): """ Get autoinputs command. corsika_autoinputs is a tool to generate random and user/host dependent parameters for CORSIKA configuration. Parameters ---------- run_number: int Run number. input_tmp_file: Path Temporary input file. Returns ------- str autoinputs command. """ corsika_bin_path = self._simtel_path.joinpath("corsika-run/corsika") log_file = self.get_file_name(file_type="log", run_number=run_number) if self._use_multipipe: log_file = log_file.with_name(f"multipipe_{log_file.name}") cmd = self._simtel_path.joinpath("sim_telarray/bin/corsika_autoinputs") cmd = str(cmd) + f" --run {corsika_bin_path}" cmd += f" -R {run_number}" cmd += ' -p "$CORSIKA_DATA"' if self._keep_seeds: logging.warning( "Using --keep-seeds option in corsika_autoinputs is not recommended. " "It should only be used for testing purposes." ) cmd += " --keep-seeds" cmd += f" {input_tmp_file} | gzip > {log_file} 2>&1" cmd += " || exit 1\n" return cmd
[docs] def get_file_name( self, simulation_software="corsika", 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() != "corsika": raise ValueError( f"simulation_software ({simulation_software}) is not supported in CorsikaRunner" ) return self.runner_service.get_file_name( file_type=file_type, run_number=run_number, mode=mode, )