From 1efeff9b40e9790a0dc7607f14caf1d796634106 Mon Sep 17 00:00:00 2001
From: Gagik Vardanyan <vardanyan@ill.fr>
Date: Thu, 13 Feb 2020 15:50:16 +0100
Subject: [PATCH] Re #27918 more flexible absolute scaling

---
 .../WorkflowAlgorithms/SANSILLReduction.py    |  73 ++++++---
 .../analysis/SANSILLAbsoluteScaleTest.py      | 144 ++++++++++++++++++
 .../tests/analysis/SANSILLReductionTest.py    |  57 -------
 .../reference/D11_AbsScaleReference.nxs.md5   |   2 +-
 .../reference/ILL_SANS_D11_IQ.nxs.md5         |   2 +-
 docs/source/release/v4.3.0/sans.rst           |   1 +
 6 files changed, 202 insertions(+), 77 deletions(-)
 create mode 100644 Testing/SystemTests/tests/analysis/SANSILLAbsoluteScaleTest.py

diff --git a/Framework/PythonInterface/plugins/algorithms/WorkflowAlgorithms/SANSILLReduction.py b/Framework/PythonInterface/plugins/algorithms/WorkflowAlgorithms/SANSILLReduction.py
index 0f13f70f04a..61ea77d2466 100644
--- a/Framework/PythonInterface/plugins/algorithms/WorkflowAlgorithms/SANSILLReduction.py
+++ b/Framework/PythonInterface/plugins/algorithms/WorkflowAlgorithms/SANSILLReduction.py
@@ -333,15 +333,6 @@ class SANSILLReduction(PythonAlgorithm):
             Processes the sample
             @param ws: input workspace
         """
-        reference_ws = self.getProperty('ReferenceInputWorkspace').value
-        coll_ws = None
-        if reference_ws:
-            if not self._check_processed_flag(reference_ws, 'Sample'):
-                self.log().warning('Reference input workspace is not processed as sample.')
-            Divide(LHSWorkspace=ws, RHSWorkspace=reference_ws, OutputWorkspace=ws, WarnOnZeroDivide=False)
-            Scale(InputWorkspace=ws, Factor=self.getProperty('WaterCrossSection').value, OutputWorkspace=ws)
-            self._mask(ws, reference_ws)
-            coll_ws = reference_ws
         sensitivity_in = self.getProperty('SensitivityInputWorkspace').value
         if sensitivity_in:
             if not self._check_processed_flag(sensitivity_in, 'Sensitivity'):
@@ -357,15 +348,61 @@ class SANSILLReduction(PythonAlgorithm):
                 DeleteWorkspace(flux_ws)
             else:
                 Divide(LHSWorkspace=ws, RHSWorkspace=flux_in, OutputWorkspace=ws, WarnOnZeroDivide=False)
-        if coll_ws:
-            self._check_distances_match(mtd[ws], coll_ws)
-            sample_coll = mtd[ws].getRun().getLogData('collimation.actual_position').value
-            ref_coll = coll_ws.getRun().getLogData('collimation.actual_position').value
-            flux_factor = (sample_coll ** 2) / (ref_coll ** 2)
-            self.log().notice('Flux factor is: ' + str(flux_factor))
-            Scale(InputWorkspace=ws, Factor=flux_factor, OutputWorkspace=ws)
-            ReplaceSpecialValues(InputWorkspace=ws, OutputWorkspace=ws,
-                                 NaNValue=0., NaNError=0., InfinityValue=0., InfinityError=0.)
+            AddSampleLog(Workspace=ws, LogText='True', LogType='String', LogName='NormalisedByFlux')
+            self._do_rescale_flux(ws, flux_in)
+        reference_ws = self.getProperty('ReferenceInputWorkspace').value
+        if reference_ws:
+            if not self._check_processed_flag(reference_ws, 'Sample'):
+                self.log().warning('Reference input workspace is not processed as sample.')
+            Divide(LHSWorkspace=ws, RHSWorkspace=reference_ws, OutputWorkspace=ws, WarnOnZeroDivide=False)
+            Scale(InputWorkspace=ws, Factor=self.getProperty('WaterCrossSection').value, OutputWorkspace=ws)
+            self._mask(ws, reference_ws)
+            self._rescale_flux(ws, reference_ws)
+        ReplaceSpecialValues(InputWorkspace=ws, OutputWorkspace=ws,
+                             NaNValue=0., NaNError=0., InfinityValue=0., InfinityError=0.)
+
+    def _rescale_flux(self, ws, ref_ws):
+        """
+            This adjusts the absolute scale after normalising by water
+            If both sample and water runs are normalised by flux, there is nothing to do
+            If one is normalised, the other is not, we log a warning
+            If neither is normalised by flux, we have to rescale by the factor
+            @param ws : the workspace to scale (sample)
+            @param ref_ws : the reference workspace (water)
+        """
+        message = 'Sample and water runs are not consistent in terms of flux normalisation; ' \
+                  'the absolute scale will not be correct. ' \
+                  'Make sure they are either both normalised or both not normalised by flux.' \
+                  'Consider specifying the sample flux also to water reduction.' \
+                  'Even if it would be at different distance, it will be rescaled correctly.'
+        run = mtd[ws].getRun()
+        run_ref = ref_ws.getRun()
+        has_log = run.hasProperty('NormalisedByFlux')
+        has_log_ref = run_ref.hasProperty('NormalisedByFlux')
+        if has_log != has_log_ref:
+            raise RuntimeError(message)
+        if has_log and has_log_ref:
+            log_val = run.getLogData('NormalisedByFlux').value
+            log_val_ref = run_ref.getLogData('NormalisedByFlux').value
+            if log_val != log_val_ref:
+                raise RuntimeError(message)
+            elif log_val == 'False':
+                self._do_rescale_flux(ws, ref_ws)
+        else:
+            self._do_rescale_flux(ws, ref_ws)
+
+    def _do_rescale_flux(self, ws, ref_ws):
+        """
+            Scales ws by the magic flux factor wrt the reference
+            @ws : input workspace to scale (sample)
+            @ref_ws : reference workspace (water)
+        """
+        self._check_distances_match(mtd[ws], ref_ws)
+        sample_l2 = mtd[ws].getRun().getLogData('L2').value
+        ref_l2 = ref_ws.getRun().getLogData('L2').value
+        flux_factor = (sample_l2 ** 2) / (ref_l2 ** 2)
+        self.log().notice('Flux factor is: ' + str(flux_factor))
+        Scale(InputWorkspace=ws, Factor=flux_factor, OutputWorkspace=ws)
 
     def _apply_absorber(self, ws, absorber_ws):
         """
diff --git a/Testing/SystemTests/tests/analysis/SANSILLAbsoluteScaleTest.py b/Testing/SystemTests/tests/analysis/SANSILLAbsoluteScaleTest.py
new file mode 100644
index 00000000000..cf39d92a6b4
--- /dev/null
+++ b/Testing/SystemTests/tests/analysis/SANSILLAbsoluteScaleTest.py
@@ -0,0 +1,144 @@
+# Mantid Repository : https://github.com/mantidproject/mantid
+#
+# Copyright &copy; 2020 ISIS Rutherford Appleton Laboratory UKRI,
+#     NScD Oak Ridge National Laboratory, European Spallation Source
+#     & Institut Laue - Langevin
+# SPDX - License - Identifier: GPL - 3.0 +
+from __future__ import (absolute_import, division, print_function)
+
+import systemtesting
+from mantid.simpleapi import *
+
+
+class D11_AbsoluteScale_Test(systemtesting.MantidSystemTest):
+
+    def __init__(self):
+        super(D11_AbsoluteScale_Test, self).__init__()
+        self.setUp()
+
+    def setUp(self):
+        config['default.facility'] = 'ILL'
+        config['default.instrument'] = 'D11'
+        config.appendDataSearchSubDir('ILL/D11/')
+
+    def cleanup(self):
+        mtd.clear()
+
+    def validate(self):
+        self.tolerance = 1e-5
+        self.tolerance_is_rel_err = True
+        self.disableChecking = ['Instrument']
+        return ['abs_scale_outputs', 'D11_AbsScaleReference.nxs']
+
+    def runTest(self):
+        # Load the mask
+        LoadNexusProcessed(Filename='D11_mask.nxs', OutputWorkspace='mask')
+
+        # Process the dark current Cd/B4C for water
+        SANSILLReduction(Run='010455', ProcessAs='Absorber', OutputWorkspace='Cdw')
+
+        # Process the empty beam for water
+        SANSILLReduction(Run='010414', ProcessAs='Beam', AbsorberInputWorkspace='Cdw', OutputWorkspace='Dbw',
+                         FluxOutputWorkspace='Flw')
+        # Water container transmission
+        SANSILLReduction(Run='010446', ProcessAs='Transmission', AbsorberInputWorkspace='Cdw',
+                         BeamInputWorkspace='Dbw', OutputWorkspace='wc_tr')
+        # Water container
+        SANSILLReduction(Run='010454', ProcessAs='Container', AbsorberInputWorkspace='Cdw',
+                         BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr', OutputWorkspace='wc')
+        # Water transmission
+        SANSILLReduction(Run='010445', ProcessAs='Transmission', AbsorberInputWorkspace='Cdw',
+                         BeamInputWorkspace='Dbw', OutputWorkspace='w_tr')
+        # Water as reference
+        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
+                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
+                         SensitivityOutputWorkspace='sens', OutputWorkspace='reference', FluxInputWorkspace='Flw')
+        # Water as sample with sensitivity and flux
+        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
+                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
+                         SensitivityInputWorkspace='sens', OutputWorkspace='water_with_sens_flux', FluxInputWorkspace='Flw')
+        # Water with itself as reference and flux
+        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
+                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
+                         ReferenceInputWorkspace='reference', OutputWorkspace='water_with_reference', FluxInputWorkspace='Flw')
+
+        # Group the worksaces
+        GroupWorkspaces(InputWorkspaces=['sens', 'reference', 'water_with_reference', 'water_with_sens_flux'],
+                        OutputWorkspace='abs_scale_outputs')
+
+
+class D11_AbsoluteScaleFlux_Test(systemtesting.MantidSystemTest):
+
+    def __init__(self):
+        super(D11_AbsoluteScaleFlux_Test, self).__init__()
+        self.setUp()
+
+    def setUp(self):
+        config['default.facility'] = 'ILL'
+        config['default.instrument'] = 'D11'
+        config.appendDataSearchSubDir('ILL/D11/')
+
+    def cleanup(self):
+        mtd.clear()
+
+    def runTest(self):
+        # Load the mask
+        LoadNexusProcessed(Filename='D11_mask.nxs', OutputWorkspace='mask')
+
+        # Calculate flux for water
+        SANSILLReduction(Run='010414', ProcessAs='Beam', OutputWorkspace='Dbw', FluxOutputWorkspace='flw')
+
+        # Reduce water with flux normalisation
+        SANSILLReduction(Run='010453', ProcessAs='Sample', MaskedInputWorkspace='mask',
+                         OutputWorkspace='water_with_flux', FluxInputWorkspace='flw')
+
+        # Reduce water without flux normalisation
+        SANSILLReduction(Run='010453', ProcessAs='Sample', MaskedInputWorkspace='mask', OutputWorkspace='water_wo_flux')
+
+        # Calculate flux for sample
+        SANSILLReduction(Run='010413', ProcessAs='Beam', OutputWorkspace='Db', FluxOutputWorkspace='fl')
+
+        # Reduce sample with flux normalisation and flux normalised water reference
+        SANSILLReduction(Run='010569', ProcessAs='Sample', MaskedInputWorkspace='mask',
+                         OutputWorkspace='sample_with_flux', FluxInputWorkspace='fl', ReferenceInputWorkspace='water_with_flux')
+
+        # Reduce sample without flux normalisation and not flux normalised water reference
+        SANSILLReduction(Run='010569', ProcessAs='Sample', MaskedInputWorkspace='mask',
+                         OutputWorkspace='sample_wo_flux', ReferenceInputWorkspace='water_wo_flux')
+
+        # Now the sample_with_flux and sample_wo_flux should be approximately at the same scale
+        result1, _ = CompareWorkspaces(Workspace1='sample_with_flux', Workspace2='sample_wo_flux', Tolerance=0.1)
+        self.assertTrue(result1)
+
+        # Then we want to simulate the situation where water has no flux measurement
+        # Reduce water, but normalise it to the sample flux (water will get scaled here)
+        SANSILLReduction(Run='010453', ProcessAs='Sample', MaskedInputWorkspace='mask',
+                         OutputWorkspace='water_with_sample_flux', FluxInputWorkspace='fl')
+
+        # Reduce sample with flux normalisation and sample flux normalised water reference
+        # Here there is no additional scaling, since both are already normalised
+        SANSILLReduction(Run='010569', ProcessAs='Sample', MaskedInputWorkspace='mask',
+                         OutputWorkspace='sample_with_flux_water_with_sample_flux',
+                         FluxInputWorkspace='fl', ReferenceInputWorkspace='water_with_sample_flux')
+
+        # Now this output should still be at the same scale as the two above
+        # (basically it is the same scaling, just happening in different place)
+        result2, _ = CompareWorkspaces(Workspace1='sample_with_flux_water_with_sample_flux',
+                                       Workspace2='sample_wo_flux', Tolerance=0.1)
+        self.assertTrue(result2)
+
+        result3, _ = CompareWorkspaces(Workspace1='sample_with_flux_water_with_sample_flux',
+                                       Workspace2='sample_with_flux', Tolerance=0.1)
+        self.assertTrue(result3)
+
+        # Finally we want to make sure that trying to divide flux normalised sample by
+        # non flux normalised water raises an error
+        kwargs = {
+            'Run' : '010569',
+            'ProcessAs' : 'Sample',
+            'MaskedInputWorkspace' : 'mask',
+            'OutputWorkspace' : 'sample_with_flux_water_wo_flux',
+            'FluxInputWorkspace' : 'fl',
+            'ReferenceInputWorkspace' : 'water_wo_flux'
+        }
+        self.assertRaises(RuntimeError, SANSILLReduction, **kwargs)
diff --git a/Testing/SystemTests/tests/analysis/SANSILLReductionTest.py b/Testing/SystemTests/tests/analysis/SANSILLReductionTest.py
index da6b2fac4a6..583a9ad25c3 100644
--- a/Testing/SystemTests/tests/analysis/SANSILLReductionTest.py
+++ b/Testing/SystemTests/tests/analysis/SANSILLReductionTest.py
@@ -299,60 +299,3 @@ class ILL_D33_Test(systemtesting.MantidSystemTest):
 
         # I(Q)
         SANSILLIntegration(InputWorkspace='sample_flux', OutputWorkspace='iq')
-
-
-class ILL_D11_AbsoluteScale_Test(systemtesting.MantidSystemTest):
-
-    def __init__(self):
-        super(ILL_D11_AbsoluteScale_Test, self).__init__()
-        self.setUp()
-
-    def setUp(self):
-        config['default.facility'] = 'ILL'
-        config['default.instrument'] = 'D11'
-        config.appendDataSearchSubDir('ILL/D11/')
-
-    def cleanup(self):
-        mtd.clear()
-
-    def validate(self):
-        self.tolerance = 1e-5
-        self.tolerance_is_rel_err = True
-        self.disableChecking = ['Instrument']
-        return ['abs_scale_outputs', 'D11_AbsScaleReference.nxs']
-
-    def runTest(self):
-        # Load the mask
-        LoadNexusProcessed(Filename='D11_mask.nxs', OutputWorkspace='mask')
-
-        # Process the dark current Cd/B4C for water
-        SANSILLReduction(Run='010455', ProcessAs='Absorber', OutputWorkspace='Cdw')
-
-        # Process the empty beam for water
-        SANSILLReduction(Run='010414', ProcessAs='Beam', AbsorberInputWorkspace='Cdw', OutputWorkspace='Dbw',
-                         FluxOutputWorkspace='Flw')
-        # Water container transmission
-        SANSILLReduction(Run='010446', ProcessAs='Transmission', AbsorberInputWorkspace='Cdw',
-                         BeamInputWorkspace='Dbw', OutputWorkspace='wc_tr')
-        # Water container
-        SANSILLReduction(Run='010454', ProcessAs='Container', AbsorberInputWorkspace='Cdw',
-                         BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr', OutputWorkspace='wc')
-        # Water transmission
-        SANSILLReduction(Run='010445', ProcessAs='Transmission', AbsorberInputWorkspace='Cdw',
-                         BeamInputWorkspace='Dbw', OutputWorkspace='w_tr')
-        # Water as reference
-        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
-                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
-                         SensitivityOutputWorkspace='sens', OutputWorkspace='reference', FluxInputWorkspace='Flw')
-        # Water as sample with sensitivity and flux
-        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
-                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
-                         SensitivityInputWorkspace='sens', OutputWorkspace='water_with_sens_flux', FluxInputWorkspace='Flw')
-        # Water with itself as reference and flux
-        SANSILLReduction(Run='010453', ProcessAs='Sample', AbsorberInputWorkspace='Cdw', MaskedInputWorkspace='mask',
-                         ContainerInputWorkspace='wc', BeamInputWorkspace='Dbw', TransmissionInputWorkspace='wc_tr',
-                         ReferenceInputWorkspace='reference', OutputWorkspace='water_with_reference', FluxInputWorkspace='Flw')
-
-        # Group the worksaces
-        GroupWorkspaces(InputWorkspaces=['sens', 'reference', 'water_with_reference', 'water_with_sens_flux'],
-                        OutputWorkspace='abs_scale_outputs')
diff --git a/Testing/SystemTests/tests/analysis/reference/D11_AbsScaleReference.nxs.md5 b/Testing/SystemTests/tests/analysis/reference/D11_AbsScaleReference.nxs.md5
index 258b1b7bb25..f54a61cae43 100644
--- a/Testing/SystemTests/tests/analysis/reference/D11_AbsScaleReference.nxs.md5
+++ b/Testing/SystemTests/tests/analysis/reference/D11_AbsScaleReference.nxs.md5
@@ -1 +1 @@
-37ca323c47478b6c49b26676a89423a8
+8d9ed6b7ed46d696b8f81b754d7c7ae7
diff --git a/Testing/SystemTests/tests/analysis/reference/ILL_SANS_D11_IQ.nxs.md5 b/Testing/SystemTests/tests/analysis/reference/ILL_SANS_D11_IQ.nxs.md5
index 1f96700f486..93a9a998fed 100644
--- a/Testing/SystemTests/tests/analysis/reference/ILL_SANS_D11_IQ.nxs.md5
+++ b/Testing/SystemTests/tests/analysis/reference/ILL_SANS_D11_IQ.nxs.md5
@@ -1 +1 @@
-cd7c39ac8e064f4a639155f3e747c832
+036c8504fc9d7f68be0f8f42c8a64e8e
diff --git a/docs/source/release/v4.3.0/sans.rst b/docs/source/release/v4.3.0/sans.rst
index 075f832984e..6a4185defd9 100644
--- a/docs/source/release/v4.3.0/sans.rst
+++ b/docs/source/release/v4.3.0/sans.rst
@@ -15,6 +15,7 @@ Improved
 - :ref:`MaskBTP <algm-MaskBTP>` now handles both old and new instrument definitions for BIOSANS and GPSANS
 - :ref:`SANSILLReduction <algm-SANSILLReduction>` and :ref:`SANSILLAutoProcess <algm-SANSILLAutoProcess>` are improved to better handle the absolute scale normalisation.
 - :ref:`SANSILLIntegration <algm-SANSILLIntegration>` will now offer to produce I(Q) for separate detector components separately, which is useful for D33.
+- :ref:`SANSILLReduction <algm-SANSILLReduction>` will now allow for correct absolute scale normalisation even in circumstances when, for example, there is no flux measurement for the water run configuration.
 - Data with invalid proton charge logs will now be fixed before performing
   slicing. A warning is emitted when this happens.
 - ISIS SANS history for top level algorithms now works correctly. A user
-- 
GitLab