Source code for data_model.data_reader
"""Helper module for reading of standardized simtools data products."""
import logging
import astropy.units as u
from astropy.io.registry.base import IORegistryError
from astropy.table import QTable
import simtools.utils.general as gen
from simtools.data_model import validate_data
from simtools.data_model.metadata_collector import MetadataCollector
__all__ = ["read_table_from_file", "read_value_from_file"]
_logger = logging.getLogger(__name__)
[docs]
def read_table_from_file(file_name, schema_file=None, validate=False, metadata_file=None):
"""
Read astropy table from file and validate against schema.
Metadata is read from metadata file or from the metadata section of the data file.
Schema for validation can be given as argument, or is determined
from the metadata associated to the file.
Parameters
----------
file_name: str or Path
Name of file to be read.
schema_file: str or Path
Name of schema file to be used for validation.
validate: bool
Validate data against schema (if true).
metadata_file: str or Path
Name of metadata file to be read.
Returns
-------
astropy Table
Table read from file.
Raises
------
FileNotFoundError
If file does not exist.
"""
try:
data_table = QTable.read(file_name)
except (FileNotFoundError, IORegistryError) as exc:
_logger.error("Error reading tabled data from %s", file_name)
raise exc
_logger.info("Reading table data from %s", file_name)
if validate:
metadata = MetadataCollector(
args_dict=None,
metadata_file_name=(metadata_file if metadata_file is not None else file_name),
data_model_name=None,
)
_validator = validate_data.DataValidator(
schema_file=(
schema_file
if schema_file is not None
else metadata.get_data_model_schema_file_name()
),
data_table=data_table,
)
return _validator.validate_and_transform()
return data_table
[docs]
def read_value_from_file(file_name, schema_file=None, validate=False):
"""
Read value from file and validate against schema.
Expect data to follow the convention for
how simulation model parameters are stored in the simulation model database: to be a single
value stored in the 'value' field (with possible units in the 'units' field).
Metadata is read from metadata file or from the metadata section of the data file.
Schema for validation can be given as argument, or is determined
from the metadata associated to the file.
Parameters
----------
file_name: str or Path
Name of file to be read.
schema_file: str or Path
Name of schema file to be used for validation.
validate: bool
Validate data against schema (if true).
Returns
-------
astro quantity or str
Value read from file. If units are given, return an astropy quantity, otherwise a string.
Return None if no value is found in the file.
Raises
------
FileNotFoundError
If file does not exist.
"""
try:
data = gen.collect_data_from_file(file_name=file_name)
except FileNotFoundError as exc:
_logger.error("Error reading data from %s", file_name)
raise exc
_logger.info("Reading data from %s", file_name)
if validate:
if schema_file is None and "meta_schema_url" in data:
schema_file = data["meta_schema_url"]
_logger.debug(f"Using schema from meta_schema_url: {schema_file}")
if schema_file is None:
_collector = MetadataCollector(None, metadata_file_name=file_name)
schema_file = _collector.get_data_model_schema_file_name()
_logger.debug(f"Using schema from meta_data_url: {schema_file}")
_validator = validate_data.DataValidator(
schema_file=schema_file,
data_dict=data,
)
data = _validator.validate_and_transform()
_logger.debug("Successful validation of yaml/json file")
_value = data.get("value")
if _value is None:
return None
_unit = data.get("unit")
return _value if _unit is None else _value * u.Unit(_unit)