Source code for ess.isissans.general

# SPDX-License-Identifier: BSD-3-Clause
# Copyright (c) 2023 Scipp contributors (https://github.com/scipp)
"""
Providers for the ISIS instruments.
"""

from typing import NewType

import sciline
import scipp as sc
import scippnexus as snx

from ess.reduce.nexus.types import NeXusTransformation, Position
from ess.sans.types import (
    BeamCenter,
    CalibratedDetector,
    CalibratedMonitor,
    CorrectForGravity,
    DetectorData,
    DetectorIDs,
    DetectorPixelShape,
    DetectorPositionOffset,
    DimsToKeep,
    Incident,
    MonitorData,
    MonitorPositionOffset,
    MonitorType,
    NeXusComponent,
    NeXusMonitorName,
    NonBackgroundWavelengthRange,
    RunNumber,
    RunTitle,
    RunType,
    SampleRun,
    ScatteringRunType,
    TofData,
    TofMonitor,
    Transmission,
    WavelengthBands,
    WavelengthMask,
)

from .io import LoadedFileContents
from .mantidio import Period


[docs] class MonitorOffset(sciline.Scope[MonitorType, sc.Variable], sc.Variable): """ Offset for monitor position for all runs. """
DetectorBankOffset = NewType('DetectorBankOffset', sc.Variable) SampleOffset = NewType('SampleOffset', sc.Variable)
[docs] def default_parameters() -> dict: return { CorrectForGravity: False, DimsToKeep: (), MonitorOffset[Incident]: MonitorOffset[Incident]( sc.vector([0, 0, 0], unit='m') ), MonitorOffset[Transmission]: MonitorOffset[Transmission]( sc.vector([0, 0, 0], unit='m') ), DetectorBankOffset: DetectorBankOffset(sc.vector([0, 0, 0], unit='m')), SampleOffset: SampleOffset(sc.vector([0, 0, 0], unit='m')), NonBackgroundWavelengthRange: None, WavelengthMask: None, WavelengthBands: None, Period: None, }
[docs] def to_detector_position_offset( global_offset: DetectorBankOffset, beam_center: BeamCenter ) -> DetectorPositionOffset[RunType]: return DetectorPositionOffset[RunType](global_offset - beam_center)
[docs] def to_monitor_position_offset( global_offset: MonitorOffset[MonitorType], ) -> MonitorPositionOffset[RunType, MonitorType]: return MonitorPositionOffset[RunType, MonitorType](global_offset)
[docs] def get_source_position( dg: LoadedFileContents[RunType], ) -> Position[snx.NXsource, RunType]: """Get source position from raw data.""" return Position[snx.NXsource, RunType](dg['data'].coords['source_position'])
[docs] def get_sample_position( dg: LoadedFileContents[RunType], offset: SampleOffset ) -> Position[snx.NXsample, RunType]: """Get sample position from raw data and apply user offset.""" return Position[snx.NXsample, RunType]( dg['data'].coords['sample_position'] + offset.to(unit='m') )
[docs] def get_detector_data( dg: LoadedFileContents[RunType], ) -> NeXusComponent[snx.NXdetector, RunType]: """Get detector data and apply user offsets to raw data. Parameters ---------- dg: Data loaded with Mantid and converted to Scipp. """ # The generic NeXus workflow will try to extract 'data' from this, which is exactly # what we also have in the Mantid data. We use the generic workflow since it also # applies offsets, etc. return NeXusComponent[snx.NXdetector, RunType](dg)
[docs] def get_calibrated_isis_detector( detector: NeXusComponent[snx.NXdetector, RunType], *, offset: DetectorPositionOffset[RunType], ) -> CalibratedDetector[RunType]: """ Replacement for :py:func:`ess.reduce.nexus.workflow.get_calibrated_detector`. Differences: - The detector position is already pre-computed. - The detector is not reshaped. The reason for the partial duplication is to avoid having to put ISIS/Mantid specific code in the generic workflow. """ da = detector['data'] position = detector['data'].coords['position'] return CalibratedDetector[RunType]( da.assign_coords(position=position + offset.to(unit=position.unit)) )
[docs] def get_monitor_data( dg: LoadedFileContents[RunType], nexus_name: NeXusMonitorName[MonitorType] ) -> NeXusComponent[MonitorType, RunType]: # The generic NeXus workflow will try to extract 'data' from this, which is exactly # what we also have in the Mantid data. We use the generic workflow since it also # applies offsets, etc. monitor = dg['monitors'][nexus_name]['data'] return NeXusComponent[MonitorType, RunType]( sc.DataGroup(data=monitor, position=monitor.coords['position']) )
[docs] def dummy_assemble_detector_data( detector: CalibratedDetector[RunType], ) -> DetectorData[RunType]: """Dummy assembly of detector data, detector already contains neutron data.""" return DetectorData[RunType](detector)
[docs] def dummy_assemble_monitor_data( monitor: CalibratedMonitor[RunType, MonitorType], ) -> MonitorData[RunType, MonitorType]: """Dummy assembly of monitor data, monitor already contains neutron data.""" return MonitorData[RunType, MonitorType](monitor)
[docs] def data_to_tof( da: DetectorData[ScatteringRunType], ) -> TofData[ScatteringRunType]: """Dummy conversion of data to time-of-flight data. The data already has a time-of-flight coordinate.""" return TofData[ScatteringRunType](da)
[docs] def monitor_to_tof( da: MonitorData[RunType, MonitorType], ) -> TofMonitor[RunType, MonitorType]: """Dummy conversion of monitor data to time-of-flight data. The monitor data already has a time-of-flight coordinate.""" return TofMonitor[RunType, MonitorType](da)
[docs] def run_number(dg: LoadedFileContents[SampleRun]) -> RunNumber: """Get the run number from the raw sample data.""" return RunNumber(int(dg['run_number']))
[docs] def run_title(dg: LoadedFileContents[SampleRun]) -> RunTitle: """Get the run title from the raw sample data.""" return RunTitle(dg['run_title'].value)
[docs] def helium3_tube_detector_pixel_shape() -> DetectorPixelShape[ScatteringRunType]: # Pixel radius and length # found here: # https://github.com/mantidproject/mantid/blob/main/instrument/SANS2D_Definition_Tubes.xml R = 0.00405 L = 0.002033984375 pixel_shape = sc.DataGroup( { 'vertices': sc.vectors( dims=['vertex'], values=[ # Coordinates in pixel-local coordinate system # Bottom face center [0, 0, 0], # Bottom face edge [R, 0, 0], # Top face center [0, L, 0], ], unit='m', ), 'nexus_class': 'NXcylindrical_geometry', } ) return pixel_shape
[docs] def lab_frame_transform() -> NeXusTransformation[snx.NXdetector, ScatteringRunType]: # Rotate +y to -x return NeXusTransformation[snx.NXdetector, ScatteringRunType]( sc.spatial.rotation(value=[0, 0, 1 / 2**0.5, 1 / 2**0.5]) )
[docs] def get_detector_ids_from_sample_run(data: TofData[SampleRun]) -> DetectorIDs: """Extract detector IDs from sample run. This overrides the function in the masking module which gets the detector IDs from the detector before loading event data. In this ISIS case files are loaded using Mantid which does not load event separately, so we get IDs from the data. """ return DetectorIDs( data.coords[ 'detector_number' if 'detector_number' in data.coords else 'detector_id' ] )
providers = ( dummy_assemble_detector_data, dummy_assemble_monitor_data, to_detector_position_offset, to_monitor_position_offset, get_source_position, get_sample_position, get_detector_data, get_calibrated_isis_detector, get_detector_ids_from_sample_run, get_monitor_data, data_to_tof, monitor_to_tof, run_number, run_title, lab_frame_transform, helium3_tube_detector_pixel_shape, )