2020-05-08 15:13:02 +02:00
|
|
|
from typing import Union
|
|
|
|
import re
|
|
|
|
import xml.etree.ElementTree as eT
|
|
|
|
import astropy.units as u
|
2020-05-29 09:36:02 +02:00
|
|
|
from ..lib.logger import logger
|
2020-05-08 15:13:02 +02:00
|
|
|
import difflib
|
|
|
|
import os
|
|
|
|
|
|
|
|
|
|
|
|
class Entry(object):
|
|
|
|
"""
|
|
|
|
A class used to represent a configuration entry.
|
|
|
|
Taken from ExoSim (https://github.com/ExoSim/ExoSimPublic)
|
|
|
|
"""
|
|
|
|
val: Union[str, bool, u.Quantity]
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
|
|
|
for key, value in kwargs.items():
|
|
|
|
self.__setattr__(key, value)
|
|
|
|
|
|
|
|
def __call__(self):
|
|
|
|
return self.val if hasattr(self, "val") else None
|
|
|
|
|
|
|
|
def parse(self, xml: eT.Element):
|
|
|
|
"""
|
|
|
|
Parse attributes of a XML element
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
xml : xml.etree.ElementTree.Element
|
|
|
|
XML element to parse the attributes from
|
|
|
|
"""
|
|
|
|
# Copy the XML attributes to object attributes
|
|
|
|
for attrib in xml.attrib.keys():
|
|
|
|
setattr(self, attrib, xml.attrib[attrib])
|
|
|
|
# parse units
|
|
|
|
attribs = list(xml.attrib.keys())
|
|
|
|
units = list(filter(re.compile(".*_unit$").match, attribs))
|
|
|
|
for unit in units:
|
|
|
|
var = unit.replace("_unit", "")
|
|
|
|
if hasattr(self, var):
|
|
|
|
try:
|
|
|
|
val = u.Quantity(list(map(float, getattr(self, var).split(','))), getattr(self, unit))
|
|
|
|
if len(val) == 1:
|
|
|
|
val = val[0]
|
|
|
|
setattr(self, var, val)
|
|
|
|
except (ValueError, LookupError):
|
2020-05-29 09:36:02 +02:00
|
|
|
logger.error("unable to convert units in entry '" + xml.tag + "': " + getattr(self, var) + " " +
|
|
|
|
getattr(self, unit), exit_=False)
|
2020-05-08 15:13:02 +02:00
|
|
|
# Convert boolean values
|
|
|
|
if hasattr(self, "val") and type(self.val) == str and self.val.lower() in ["false", "true"]:
|
|
|
|
self.val = (self.val.lower() == "true")
|
|
|
|
|
2020-05-26 22:10:41 +02:00
|
|
|
def check_quantity(self, name: str, unit: u.Unit, use_default: bool = True) -> Union[None, str]:
|
2020-05-08 17:05:17 +02:00
|
|
|
"""
|
|
|
|
Check a parameter as type quantity
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name : str
|
|
|
|
The name of the parameter to be checked.
|
|
|
|
unit : Quantity
|
|
|
|
The default quantity to be used for conversion and equality checking.
|
2020-05-29 09:36:02 +02:00
|
|
|
use_default : bool
|
|
|
|
Use the given unit as default unit and try to convert strings to Quantities with this unit.
|
2020-05-08 17:05:17 +02:00
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mes : Union[None, str]
|
|
|
|
The error message of the check. This will be None if the check was successful.
|
|
|
|
"""
|
2020-05-08 15:13:02 +02:00
|
|
|
if not hasattr(self, name):
|
|
|
|
return "Parameter '" + name + "' not found."
|
|
|
|
attr = getattr(self, name)
|
|
|
|
if type(attr) != u.Quantity:
|
2020-05-26 22:10:41 +02:00
|
|
|
if unit == u.dimensionless_unscaled or use_default:
|
2020-05-08 15:13:02 +02:00
|
|
|
try:
|
2020-05-26 22:10:41 +02:00
|
|
|
self.__setattr__(name, float(attr) * unit)
|
2020-05-08 15:13:02 +02:00
|
|
|
except ValueError:
|
|
|
|
return "Expected parameter '" + name + "' with unit '" + unit.to_string() + \
|
|
|
|
"' but got no unit and cannot convert '" + attr + "' to a numeric value."
|
|
|
|
else:
|
|
|
|
return "Expected parameter '" + name + "' with unit '" + unit.to_string() + "' but got no unit."
|
2020-09-30 21:23:23 +02:00
|
|
|
elif not attr.unit.is_equivalent(unit):
|
2020-07-20 16:27:59 +02:00
|
|
|
if unit == u.K and attr.unit == u.Celsius:
|
2020-07-22 11:07:02 +02:00
|
|
|
setattr(self, name, attr.to(unit, equivalencies=u.temperature()))
|
|
|
|
setattr(self, name + '_unit', unit.to_string())
|
|
|
|
elif unit.is_equivalent(u.m) and attr.unit.is_equivalent(u.Hz):
|
|
|
|
setattr(self, name, attr.to(unit, equivalencies=u.spectral()))
|
|
|
|
setattr(self, name + '_unit', unit.to_string())
|
2020-07-20 16:27:59 +02:00
|
|
|
else:
|
|
|
|
return "Expected parameter '" + name + "' with unit equivalent to '" + unit.to_string() + \
|
2020-05-08 15:13:02 +02:00
|
|
|
"' but got unit '" + attr.unit.to_string() + "'."
|
|
|
|
return None
|
|
|
|
|
2020-05-20 09:12:55 +02:00
|
|
|
def check_selection(self, name: str, choices: list) -> Union[None, str]:
|
2020-05-08 17:05:17 +02:00
|
|
|
"""
|
|
|
|
Check a parameter against a list of possible choices. In case of a mismatch, a recommendation will be given.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name : str
|
|
|
|
The name of the parameter to be checked.
|
|
|
|
choices : list
|
|
|
|
List of choices to be used for checking.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mes : Union[None, str]
|
|
|
|
The error message of the check. This will be None if the check was successful.
|
|
|
|
"""
|
2020-05-08 15:13:02 +02:00
|
|
|
if not hasattr(self, name):
|
|
|
|
return "Parameter '" + name + "' not found."
|
|
|
|
attr = getattr(self, name)
|
|
|
|
if type(attr) != str:
|
|
|
|
return "Expected parameter '" + name + "' to be of type string."
|
|
|
|
if attr not in choices:
|
2020-05-11 10:37:23 +02:00
|
|
|
match = difflib.get_close_matches(attr, choices, 1)
|
|
|
|
if len(match) > 0:
|
|
|
|
# noinspection PyTypeChecker
|
2020-05-29 09:36:02 +02:00
|
|
|
return "Value '" + attr + "' not allowed for parameter '" + name + "'. Did you mean '" + \
|
2020-05-11 10:37:23 +02:00
|
|
|
match[0] + "'?"
|
|
|
|
else:
|
|
|
|
return "Value '" + attr + "' not allowed for parameter '" + name + "'."
|
2020-05-08 15:13:02 +02:00
|
|
|
return None
|
|
|
|
|
|
|
|
def check_file(self, name) -> Union[None, str]:
|
2020-05-08 17:05:17 +02:00
|
|
|
"""
|
|
|
|
Check a parameter to be a valid path to a file.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name : str
|
|
|
|
The name of the parameter to be checked.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mes : Union[None, str]
|
|
|
|
The error message of the check. This will be None if the check was successful.
|
|
|
|
"""
|
2020-05-08 15:13:02 +02:00
|
|
|
if not hasattr(self, name):
|
|
|
|
return "Parameter '" + name + "' not found."
|
|
|
|
if not os.path.isfile(getattr(self, name)):
|
|
|
|
return "File '" + getattr(self, name) + "' does not exist."
|
|
|
|
|
2020-05-18 10:56:38 +02:00
|
|
|
def check_path(self, name) -> Union[None, str]:
|
|
|
|
"""
|
|
|
|
Check a parameter to be a valid path to a file.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name : str
|
|
|
|
The name of the parameter to be checked.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mes : Union[None, str]
|
|
|
|
The error message of the check. This will be None if the check was successful.
|
|
|
|
"""
|
|
|
|
if not hasattr(self, name):
|
|
|
|
return "Parameter '" + name + "' not found."
|
|
|
|
if not os.path.isdir(getattr(self, name)):
|
|
|
|
return "Path '" + getattr(self, name) + "' does not exist."
|
|
|
|
|
2020-05-08 15:13:02 +02:00
|
|
|
def check_float(self, name) -> Union[None, str]:
|
2020-05-08 17:05:17 +02:00
|
|
|
"""
|
|
|
|
Check a parameter to be a floating point value
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
name : str
|
|
|
|
The name of the parameter to be checked.
|
|
|
|
|
|
|
|
Returns
|
|
|
|
-------
|
|
|
|
mes : Union[None, str]
|
|
|
|
The error message of the check. This will be None if the check was successful.
|
|
|
|
"""
|
2020-05-08 15:13:02 +02:00
|
|
|
if not hasattr(self, name):
|
|
|
|
return "Parameter '" + name + "' not found."
|
|
|
|
attr = getattr(self, name)
|
|
|
|
if type(attr) == float:
|
|
|
|
return None
|
|
|
|
elif type(attr) == u.Quantity:
|
|
|
|
setattr(self, name, attr.value)
|
|
|
|
elif type(attr) == str:
|
|
|
|
try:
|
|
|
|
setattr(self, name, float(attr))
|
|
|
|
except ValueError:
|
|
|
|
return "Cannot convert parameter '" + name + "' with value '" + attr + "' to a numeric value."
|
|
|
|
elif type(attr) == int:
|
|
|
|
setattr(self, name, float(attr))
|
|
|
|
else:
|
|
|
|
return "Expected parameter '" + name + "' to be numeric but got '" + type(attr) + "' instead."
|