From f8dc1be57cac3072eaee28a617ec8950ba4f276a Mon Sep 17 00:00:00 2001 From: David Fairbrother <DavidFair@users.noreply.github.com> Date: Mon, 20 Mar 2017 18:23:37 +0000 Subject: [PATCH] Re #19156 Added unit tests for RunDetails --- .../ISISPowderRunDetailsTest.yaml.md5 | 1 + .../ISISPowderRunDetailsTestCallable.yaml.md5 | 1 + scripts/CMakeLists.txt | 1 + .../isis_powder/routines/RunDetails.py | 20 +-- scripts/test/ISISPowderRunDetailsTest.py | 127 ++++++++++++++++++ 5 files changed, 141 insertions(+), 9 deletions(-) create mode 100644 Testing/Data/UnitTest/ISISPowderRunDetailsTest.yaml.md5 create mode 100644 Testing/Data/UnitTest/ISISPowderRunDetailsTestCallable.yaml.md5 create mode 100644 scripts/test/ISISPowderRunDetailsTest.py diff --git a/Testing/Data/UnitTest/ISISPowderRunDetailsTest.yaml.md5 b/Testing/Data/UnitTest/ISISPowderRunDetailsTest.yaml.md5 new file mode 100644 index 00000000000..7279eca1d3e --- /dev/null +++ b/Testing/Data/UnitTest/ISISPowderRunDetailsTest.yaml.md5 @@ -0,0 +1 @@ +1c4e481dc080a73039743d7e2fdb461f diff --git a/Testing/Data/UnitTest/ISISPowderRunDetailsTestCallable.yaml.md5 b/Testing/Data/UnitTest/ISISPowderRunDetailsTestCallable.yaml.md5 new file mode 100644 index 00000000000..a46f33621ee --- /dev/null +++ b/Testing/Data/UnitTest/ISISPowderRunDetailsTestCallable.yaml.md5 @@ -0,0 +1 @@ +3e14c49a4d8691f3fe7fe92a95dfc8f3 diff --git a/scripts/CMakeLists.txt b/scripts/CMakeLists.txt index 10eca015430..941d1418b9a 100644 --- a/scripts/CMakeLists.txt +++ b/scripts/CMakeLists.txt @@ -47,6 +47,7 @@ set ( TEST_PY_FILES test/ISISPowderAbsorptionTest.py test/ISISPowderCommonTest.py test/ISISPowderInstrumentSettingsTest.py + test/ISISPowderRunDetailsTest.py test/ISISPowderYamlParserTest.py test/PyChopTest.py test/ReductionSettingsTest.py diff --git a/scripts/Diffraction/isis_powder/routines/RunDetails.py b/scripts/Diffraction/isis_powder/routines/RunDetails.py index c1e07a8417a..571e69b3dbb 100644 --- a/scripts/Diffraction/isis_powder/routines/RunDetails.py +++ b/scripts/Diffraction/isis_powder/routines/RunDetails.py @@ -18,15 +18,17 @@ def create_run_details_object(run_number_string, inst_settings, is_vanadium_run, # Always make sure the offset file name is included if splined_name_list: - splined_name_list.append(offset_file_name) + # Force Python to make a copy so we don't modify original + new_splined_list = list(splined_name_list) + new_splined_list.append(offset_file_name) else: - splined_name_list = [offset_file_name] + new_splined_list = [offset_file_name] # These can either be generic or custom so defer to another method results_dict = _get_customisable_attributes( cal_dict=cal_map_dict, inst_settings=inst_settings, empty_run_call=empty_run_call, grouping_name_call=grouping_file_name_call, vanadium_run_call=vanadium_run_call, - splined_name_list=splined_name_list) + splined_name_list=new_splined_list) vanadium_run_string = results_dict["vanadium_runs"] @@ -37,7 +39,6 @@ def create_run_details_object(run_number_string, inst_settings, is_vanadium_run, else: output_run_string = run_number_string - # Sample empty if there is one sample_empty = inst_settings.sample_empty if hasattr(inst_settings, "sample_empty") else None @@ -104,7 +105,11 @@ class RunDetailsFuncWrapper(object): # Holds a callable method, associated args and return value so we can pass it in # as a single method - def __init__(self, function=None, func_kwargs=None): + def __init__(self, function=None, *args, **func_kwargs): + if args: + # If we allow args with position Python gets confused as the forwarded value can be in the first place too + raise RuntimeError("Cannot use un-named arguments with callable methods") + self.function = function self.function_kwargs = func_kwargs @@ -136,11 +141,8 @@ class RunDetailsFuncWrapper(object): self._previous_callable = previous_callable def add_to_func_chain(self, function, *args, **func_kwargs): - if args: - # If we allow args with position Python gets confused as the forwarded value can be in the first place too - raise RuntimeError("Cannot use un-named arguments with callable methods") # Construct a new object that will be the next in line - next_in_chain = RunDetailsFuncWrapper(function=function, func_kwargs=func_kwargs) + next_in_chain = RunDetailsFuncWrapper(function=function, *args, **func_kwargs) next_in_chain._set_previous_callable(self) return next_in_chain diff --git a/scripts/test/ISISPowderRunDetailsTest.py b/scripts/test/ISISPowderRunDetailsTest.py new file mode 100644 index 00000000000..b61c78328b0 --- /dev/null +++ b/scripts/test/ISISPowderRunDetailsTest.py @@ -0,0 +1,127 @@ +from __future__ import (absolute_import, division, print_function) + +import mantid.api +import tempfile +import os +import random +import string +import unittest +import warnings + +from isis_powder.routines import RunDetails + + +class ISISPowderInstrumentRunDetailsTest(unittest.TestCase): + + def setup_mock_inst_settings(self, yaml_file_path): + calibration_dir = tempfile.mkdtemp() + self._folders_to_remove = [calibration_dir] + + test_configuration_path = mantid.api.FileFinder.getFullPath(yaml_file_path) + if not test_configuration_path or len(test_configuration_path) <= 0: + self.fail("Could not find the unit test input file called: " + str(yaml_file_path)) + mock_inst = MockInstSettings(cal_file_path=test_configuration_path, calibration_dir=calibration_dir) + return mock_inst + + def tearDown(self): + for folder in self._folders_to_remove: + try: + os.rmdir(folder) + except OSError: + warnings.warn("Could not remove the folder at the following path:\n" + str(folder)) + + def test_create_run_details_object(self): + # These attributes are based on a flat YAML file at the specified path + expected_label = "16_4" + expected_vanadium_runs = "11-12" + expected_empty_runs = "13-14" + expected_offset_file_name = "offset_file_name" + run_number_string = "17-18" + mock_inst = self.setup_mock_inst_settings(yaml_file_path="ISISPowderRunDetailsTest.yaml") + + output_obj = RunDetails.create_run_details_object(run_number_string=run_number_string, inst_settings=mock_inst, + is_vanadium_run=False) + + self.assertEqual(output_obj.empty_runs, expected_empty_runs) + self.assertEqual(output_obj.grouping_file_path, + os.path.join(mock_inst.calibration_dir, mock_inst.grouping_file_name)) + self.assertEqual(output_obj.label, expected_label) + self.assertEqual(output_obj.offset_file_path, + os.path.join(mock_inst.calibration_dir, expected_label, expected_offset_file_name)) + self.assertEqual(output_obj.output_run_string, run_number_string) + self.assertEqual(output_obj.run_number, 17) + self.assertEqual(output_obj.vanadium_run_numbers, expected_vanadium_runs) + + def test_create_run_details_object_when_van_cal(self): + # When we are running the vanadium calibration we expected the run number to take the vanadium + # number instead + run_number_string = "17-18" + expected_vanadium_runs = "11-12" + mock_inst = self.setup_mock_inst_settings(yaml_file_path="ISISPowderRunDetailsTest.yaml") + output_obj = RunDetails.create_run_details_object(run_number_string=run_number_string, inst_settings=mock_inst, + is_vanadium_run=True) + + self.assertEqual(expected_vanadium_runs, output_obj.run_number) + self.assertEqual(output_obj.vanadium_run_numbers, output_obj.run_number) + self.assertEqual(expected_vanadium_runs, output_obj.output_run_string) + + def test_callable_params_are_used(self): + # These attributes are based on a custom YAML file at the specified path + expected_label = "16_4" + expected_vanadium_runs = "11-12" + expected_empty_runs = "13-14" + expected_grouping_file_name = "grouping_file" + run_number_string = "17-18" + mock_inst = self.setup_mock_inst_settings(yaml_file_path="ISISPowderRunDetailsTestCallable.yaml") + + # Get the YAML file as a dict first + wrapped_funcs = RunDetails.WrappedFunctionsRunDetails + + yaml_callable = RunDetails.RunDetailsFuncWrapper(function=wrapped_funcs.get_cal_mapping_dict, + run_number_string=run_number_string, inst_settings=mock_inst) + + empty_callable = yaml_callable.add_to_func_chain(function=wrapped_funcs.cal_dictionary_key_helper, + key="custom_empty_run_numbers") + vanadium_callable = yaml_callable.add_to_func_chain(function=wrapped_funcs.cal_dictionary_key_helper, + key="custom_vanadium_run_numbers") + grouping_callable = RunDetails.RunDetailsFuncWrapper(function=lambda: expected_grouping_file_name) + + output_obj = RunDetails.create_run_details_object(run_number_string=run_number_string, inst_settings=mock_inst, + is_vanadium_run=True, empty_run_call=empty_callable, + vanadium_run_call=vanadium_callable, + grouping_file_name_call=grouping_callable) + + self.assertEqual(output_obj.label, expected_label) + self.assertEqual(output_obj.empty_runs, expected_empty_runs) + self.assertEqual(output_obj.grouping_file_path, + os.path.join(mock_inst.calibration_dir, expected_grouping_file_name)) + self.assertEqual(output_obj.vanadium_run_numbers, expected_vanadium_runs) + self.assertEqual(output_obj.run_number, expected_vanadium_runs) + + def test_run_details_splined_name_list_is_used(self): + expected_vanadium_runs = "11-12" + expected_offset_file_name = "offset_file_name" + splined_name_list = ["bar", "bang", "baz"] + mock_inst = self.setup_mock_inst_settings(yaml_file_path="ISISPowderRunDetailsTest.yaml") + output_obj = RunDetails.create_run_details_object(run_number_string=10, inst_settings=mock_inst, + is_vanadium_run=False, splined_name_list=splined_name_list) + + expected_splined_out_str = ''.join('_' + val for val in splined_name_list) + expected_output_name = "VanSplined_" + expected_vanadium_runs + expected_splined_out_str + expected_output_name += '_' + expected_offset_file_name + ".nxs" + expected_path = os.path.join(mock_inst.calibration_dir, output_obj.label, expected_output_name) + self.assertEqual(expected_path, output_obj.splined_vanadium_file_path) + + +class MockInstSettings(object): + def __init__(self, cal_file_path, calibration_dir): + self.calibration_dir = calibration_dir + self.cal_mapping_path = cal_file_path + self.grouping_file_name = MockInstSettings.gen_random_string() + + @staticmethod + def gen_random_string(): + return ''.join(random.choice(string.lowercase) for _ in range(10)) + +if __name__ == "__main__": + unittest.main() -- GitLab