"""Application configuration."""
import argparse
import logging
import os
import sys
import uuid
import astropy.units as u
from dotenv import load_dotenv
import simtools.configuration.commandline_parser as argparser
from simtools.db.db_handler import jsonschema_db_dict
from simtools.io_operations import io_handler
from simtools.utils import general as gen
__all__ = [
"Configurator",
"InvalidConfigurationParameterError",
]
[docs]
class InvalidConfigurationParameterError(Exception):
"""Exception for Invalid configuration parameter."""
[docs]
class Configurator:
"""
Application configuration.
Allow to set configuration parameters by
- command line arguments
- configuration file (yml file)
- configuration dict when calling the class
- environmental variables
Assigns unique ACTIVITY_ID to this configuration (uuid).
Configuration parameter names are converted always to lower case.
Parameters
----------
config: dict
Configuration parameters as dict.
label: str
Class label.
usage: str
Application usage description.
description: str
Text displayed as description.
epilog: str
Text display after all arguments.
"""
def __init__(self, config=None, label=None, usage=None, description=None, epilog=None):
"""Initialize Configurator."""
self._logger = logging.getLogger(__name__)
self._logger.debug("Init Configuration")
self.config_class_init = config
self.label = label
self.config = {}
self.parser = argparser.CommandLineParser(
prog=self.label,
usage=usage,
description=description,
epilog=epilog,
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
[docs]
def default_config(self, arg_list=None, add_db_config=False):
"""
Return dictionary of default configuration.
Parameters
----------
arg_list: list
List of arguments.
add_db_config: bool
Add DB configuration file.
Returns
-------
dict
Configuration parameters as dict.
"""
self.parser.initialize_default_arguments()
simulation_model = None
if arg_list and "--site" in arg_list:
simulation_model = ["site"]
if arg_list and "--telescope" in arg_list:
simulation_model = ["site", "telescope"]
self.parser.initialize_simulation_model_arguments(simulation_model)
if add_db_config:
self.parser.initialize_db_config_arguments()
self._fill_config(arg_list)
return self.config
[docs]
def initialize(
self,
require_command_line=True,
paths=True,
output=False,
simulation_model=None,
simulation_configuration=None,
db_config=False,
job_submission=False,
):
"""
Initialize application configuration.
Configure from command line, configuration file, class config, or environmental variable.
Priorities in parameter settings.
1. command line; 2. yaml file; 3. class init; 4. env variables.
Conflicting configuration settings raise an Exception, with the exception of settings \
from environmental variables, which are only done when the configuration parameter \
is None.
Parameters
----------
require_command_line: bool
Require at least one command line argument.
paths: bool
Add path configuration to list of args.
output: bool
Add output file configuration to list of args.
simulation_model: list
List of simulation model configuration parameters to add to list of args
simulation_configuration: dict
Dict of simulation software configuration parameters to add to list of args.
db_config: bool
Add database configuration parameters to list of args.
job_submission: bool
Add job submission configuration to list of args.
Returns
-------
dict
Configuration parameters as dict.
dict
Dictionary with DB parameters
"""
self.parser.initialize_default_arguments(
paths=paths,
output=output,
simulation_model=simulation_model,
simulation_configuration=simulation_configuration,
db_config=db_config,
job_submission=job_submission,
)
self._fill_from_command_line(require_command_line=require_command_line)
self._fill_from_config_file(self.config.get("config"))
self._fill_from_config_dict(self.config_class_init)
self._fill_from_environmental_variables()
if self.config.get("activity_id", None) is None:
self.config["activity_id"] = str(uuid.uuid4())
if self.config["label"] is None:
self.config["label"] = self.label
self._initialize_io_handler()
if output:
self._initialize_output()
_db_dict = self._get_db_parameters()
return self.config, _db_dict
def _fill_from_command_line(self, arg_list=None, require_command_line=True):
"""
Fill configuration parameters from command line arguments.
Triggers a print of the help if no command line arguments are given and
require_command_line is set.
Parameters
----------
arg_list: list
List of arguments.
require_command_line: bool
Require at least one command line argument.
"""
if arg_list is None:
arg_list = sys.argv[1:]
if require_command_line and len(arg_list) == 0:
self._logger.debug("No command line arguments given, printing help.")
arg_list = ["--help"]
if "--config" in arg_list:
self._reset_required_arguments()
self._fill_config(arg_list)
def _reset_required_arguments(self):
"""
Reset required parser arguments (i.e., arguments added with "required=True").
Includes also mutually exclusive groups.
Access protected attributes of parser (no public method available).
"""
for group in self.parser._mutually_exclusive_groups: # pylint: disable=protected-access
group.required = False
for action in self.parser._actions: # pylint: disable=protected-access
action.required = False
def _fill_from_config_dict(self, input_dict, overwrite=False):
"""
Fill configuration parameters from dictionary.
Enforce that configuration parameter names are lower case.
Parameters
----------
input_dict: dict
dictionary with configuration parameters.
overwrite: bool
overwrite existing configuration parameters.
"""
_tmp_config = {}
try:
for key, value in input_dict.items():
if not overwrite:
self._check_parameter_configuration_status(key, value)
_tmp_config[key.lower()] = value
except AttributeError:
pass
self._fill_config(_tmp_config)
def _check_parameter_configuration_status(self, key, value):
"""
Check if a parameter is already configured and not still set to the default value.
Allow configuration with None values.
Parameters
----------
key, value
parameter key, value to be checked
Raises
------
InvalidConfigurationParameterError
if parameter has already been defined with a different value.
"""
# parameter not changed or None
if self.parser.get_default(key) == self.config[key] or self.config[key] is None:
return
# parameter already set
if key in self.config and self.config[key] != value:
self._logger.error(
f"Inconsistent configuration parameter ({key}) definition "
f"({self.config[key]} vs {value})"
)
raise InvalidConfigurationParameterError
def _fill_from_config_file(self, config_file):
"""
Read and fill configuration parameters from yaml file.
Parameters
----------
config file: str
Name of configuration file name
Raises
------
FileNotFoundError
if configuration file has not been found.
"""
try:
self._logger.debug(f"Reading configuration from {config_file}")
_config_dict = (
gen.collect_data_from_file(file_name=config_file) if config_file else None
)
# yaml parser adds \n in multiline strings, remove them
_config_dict = gen.remove_substring_recursively_from_dict(_config_dict, substring="\n")
if "CTA_SIMPIPE" in _config_dict:
try:
self._fill_from_config_dict(
input_dict=gen.change_dict_keys_case(
_config_dict["CTA_SIMPIPE"]["CONFIGURATION"],
),
overwrite=True,
)
except KeyError:
self._logger.info(f"No CTA_SIMPIPE:CONFIGURATION dict found in {config_file}.")
else:
self._fill_from_config_dict(
input_dict=gen.change_dict_keys_case(_config_dict), overwrite=True
)
# TypeError is raised for config_file=None
except (TypeError, AttributeError):
pass
except FileNotFoundError:
self._logger.error(f"Configuration file not found: {config_file}")
raise
def _fill_from_environmental_variables(self):
"""
Fill any configuration parameters from environmental variables or from file (e.g., ".env").
Only parameters which are not already configured are changed (i.e., parameter is None).
"""
_env_dict = {}
try:
load_dotenv(self.config["env_file"])
except KeyError:
pass
for key, value in self.config.items():
# environmental variables for simtools should always start with SIMTOOLS_
env_variable_to_read = f"SIMTOOLS_{key.upper()}"
if value is None:
env_value = os.environ.get(env_variable_to_read)
if env_value is not None:
env_value = env_value.split("#")[0].strip().replace('"', "").replace("'", "")
_env_dict[key] = env_value
self._fill_from_config_dict(_env_dict)
def _initialize_io_handler(self):
"""Initialize IOHandler with input and output paths."""
_io_handler = io_handler.IOHandler()
_io_handler.set_paths(
output_path=self.config.get("output_path", None),
use_plain_output_path=self.config.get("use_plain_output_path", False),
data_path=self.config.get("data_path", None),
model_path=self.config.get("model_path", None),
)
def _initialize_output(self):
"""Initialize default output file names (in case output_file is not configured)."""
if self.config.get("output_file", None) is None:
self.config["output_file_from_default"] = True
prefix = "TEST"
label = extension = ""
if not self.config.get("test", False):
prefix = self.config["activity_id"]
if self.config.get("label", "") and len(self.config.get("label", "")) > 0:
label = f"-{self.config['label']}"
if len(self.config.get("output_file_format", "")) > 0:
extension = f".{self.config['output_file_format']}"
self.config["output_file"] = f"{prefix}{label}{extension}"
@staticmethod
def _arglist_from_config(input_var):
"""
Convert input list of strings as needed by argparse.
Special cases:
- lists as arguments (using e.g., nargs="+") are expanded
- boolean are expected to be handled as action="store_true" or "store_false"
- None values or zero length values are ignored (this means setting a parameter
to none or "" is not allowed).
Ignore values which are None or of zero length.
Parameters
----------
input_var: dict, list, None
Dictionary/list of commands to convert to list.
Returns
-------
list
Dict keys and values as dict.
"""
if isinstance(input_var, dict):
_list_args = []
for key, value in input_var.items():
if isinstance(value, list):
_list_args.append("--" + key)
_list_args.extend(map(str, value))
elif isinstance(value, u.Quantity) or (
not isinstance(value, bool) and value is not None and len(str(value)) > 0
):
_list_args.append("--" + key)
_list_args.append(str(value))
elif value:
_list_args.append("--" + key)
return _list_args
try:
return [str(value) for value in list(input_var) if value != "None"]
except TypeError:
return []
@staticmethod
def _convert_string_none_to_none(input_dict):
"""
Convert string type 'None' to type None (argparse returns None as str).
Parameters
----------
input_dict
Dictionary with values to be converted.
"""
for key, value in input_dict.items():
input_dict[key] = None if value == "None" else value
return input_dict
def _fill_config(self, input_container):
"""
Fill configuration dictionary.
Parameters
----------
input_container
List or dictionary with configuration updates.
"""
self.config = self._convert_string_none_to_none(
vars(
self.parser.parse_args(
self._arglist_from_config(self.config)
+ self._arglist_from_config(input_container)
)
)
)
def _get_db_parameters(self):
"""
Return parameters for DB configuration.
Returns
-------
dict
Dictionary with DB parameters.
"""
db_params = jsonschema_db_dict["properties"].keys()
return {param: self.config.get(param) for param in db_params if param in self.config}