Source code for testing.validate_output

"""Compare application output to reference output."""

import logging
from pathlib import Path

import numpy as np
from astropy.table import Table

import simtools.utils.general as gen
from simtools.testing import assertions

_logger = logging.getLogger(__name__)


[docs] def validate_all_tests(config, request, config_file_model_version): """ Validate test output for all integration tests. Parameters ---------- config: dict Integration test configuration dictionary. request: request Request object. config_file_model_version: str Model version from the configuration file. """ if request.config.getoption("--model_version") is None: validate_application_output(config) elif config_file_model_version is not None: _from_command_line = request.config.getoption("--model_version") _from_config_file = config_file_model_version if _from_command_line == _from_config_file: validate_application_output(config)
[docs] def validate_application_output(config): """ Validate application output against expected output. Expected output is defined in configuration file. Parameters ---------- config: dict dictionary with the configuration for the application test. """ if "INTEGRATION_TESTS" not in config: return for integration_test in config["INTEGRATION_TESTS"]: _logger.info(f"Testing application output: {integration_test}") if "REFERENCE_OUTPUT_FILE" in integration_test: _validate_reference_output_file(config, integration_test) if "TEST_OUTPUT_FILES" in integration_test: _validate_output_path_and_file(config, integration_test["TEST_OUTPUT_FILES"]) if "OUTPUT_FILE" in integration_test: _validate_output_path_and_file( config, [{"PATH_DESCRIPTOR": "OUTPUT_PATH", "FILE": integration_test["OUTPUT_FILE"]}], ) if "FILE_TYPE" in integration_test: assert assertions.assert_file_type( integration_test["FILE_TYPE"], Path(config["CONFIGURATION"]["OUTPUT_PATH"]).joinpath( config["CONFIGURATION"]["OUTPUT_FILE"] ), )
def _validate_reference_output_file(config, integration_test): """Compare with reference output file.""" assert compare_files( integration_test["REFERENCE_OUTPUT_FILE"], Path(config["CONFIGURATION"]["OUTPUT_PATH"]).joinpath( config["CONFIGURATION"]["OUTPUT_FILE"] ), integration_test.get("TOLERANCE", 1.0e-5), integration_test.get("TEST_COLUMNS", None), ) def _validate_output_path_and_file(config, integration_file_tests): """Check if output paths and files exist.""" for file_test in integration_file_tests: try: output_path = config["CONFIGURATION"][file_test["PATH_DESCRIPTOR"]] except KeyError as exc: raise KeyError( f"Path {file_test['PATH_DESCRIPTOR']} not found in integration test configuration." ) from exc output_file_path = Path(output_path) / file_test["FILE"] _logger.info(f"Checking path: {output_file_path}") assert output_file_path.exists() if "EXPECTED_OUTPUT" in file_test: assert assertions.check_output_from_sim_telarray( output_file_path, file_test["EXPECTED_OUTPUT"], )
[docs] def compare_files(file1, file2, tolerance=1.0e-5, test_columns=None): """ Compare two files of file type ecsv, json or yaml. Parameters ---------- file1: str First file to compare file2: str Second file to compare tolerance: float Tolerance for comparing numerical values. test_columns: list List of columns to compare. If None, all columns are compared. Returns ------- bool True if the files are equal. """ _file1_suffix = Path(file1).suffix _file2_suffix = Path(file2).suffix if _file1_suffix != _file2_suffix: raise ValueError(f"File suffixes do not match: {file1} and {file2}") if _file1_suffix == ".ecsv": return compare_ecsv_files(file1, file2, tolerance, test_columns) if _file1_suffix in (".json", ".yaml", ".yml"): return compare_json_or_yaml_files(file1, file2) _logger.warning(f"Unknown file type for files: {file1} and {file2}") return False
[docs] def compare_json_or_yaml_files(file1, file2, tolerance=1.0e-2): """ Compare two json or yaml files. Take into account float comparison for sim_telarray string-embedded floats. Parameters ---------- file1: str First file to compare file2: str Second file to compare tolerance: float Tolerance for comparing numerical values. Returns ------- bool True if the files are equal. """ data1 = gen.collect_data_from_file(file1) data2 = gen.collect_data_from_file(file2) _logger.debug(f"Comparing json/yaml files: {file1} and {file2}") if data1 == data2: return True if "value" in data1 and isinstance(data1["value"], str): value_list_1 = gen.convert_string_to_list(data1.pop("value")) value_list_2 = gen.convert_string_to_list(data2.pop("value")) return np.allclose(value_list_1, value_list_2, rtol=tolerance) return data1 == data2
[docs] def compare_ecsv_files(file1, file2, tolerance=1.0e-5, test_columns=None): """ Compare two ecsv files. The comparison is successful if: - same number of rows - numerical values in columns are close The comparison can be restricted to a subset of columns with some additional cuts applied. This is configured through the test_columns parameter. This is a list of dictionaries, where each dictionary contains the following key-value pairs: - TEST_COLUMN_NAME: column name to compare. - CUT_COLUMN_NAME: column for filtering. - CUT_CONDITION: condition for filtering. Parameters ---------- file1: str First file to compare file2: str Second file to compare tolerance: float Tolerance for comparing numerical values. test_columns: list List of columns to compare. If None, all columns are compared. """ _logger.info(f"Comparing files: {file1} and {file2}") table1 = Table.read(file1, format="ascii.ecsv") table2 = Table.read(file2, format="ascii.ecsv") if test_columns is None: test_columns = [{"TEST_COLUMN_NAME": col} for col in table1.colnames] def generate_mask(table, column, condition): """Generate a boolean mask based on the condition (note the usage of eval).""" return ( eval(f"table['{column}'] {condition}") # pylint: disable=eval-used if condition else np.ones(len(table), dtype=bool) ) for col_dict in test_columns: col_name = col_dict["TEST_COLUMN_NAME"] mask1 = generate_mask( table1, col_dict.get("CUT_COLUMN_NAME", ""), col_dict.get("CUT_CONDITION", "") ) mask2 = generate_mask( table2, col_dict.get("CUT_COLUMN_NAME", ""), col_dict.get("CUT_CONDITION", "") ) table1_masked, table2_masked = table1[mask1], table2[mask2] if len(table1_masked) != len(table2_masked): return False if np.issubdtype(table1_masked[col_name].dtype, np.floating): if not np.allclose(table1_masked[col_name], table2_masked[col_name], rtol=tolerance): _logger.warning(f"Column {col_name} outside of relative tolerance {tolerance}") return False return True