Source code for ess.nmx.reduction

# SPDX-License-Identifier: BSD-3-Clause
# Copyright (c) 2023 Scipp contributors (https://github.com/scipp)
import scipp as sc

from .types import (
    MaximumTimeOfArrival,
    McStasWeight2CountScaleFactor,
    MinimumTimeOfArrival,
    NMXDetectorMetadata,
    NMXExperimentMetadata,
    NMXReducedCounts,
    NMXReducedDataGroup,
    NMXReducedProbability,
    PixelIds,
    ProtonCharge,
    RawEventProbability,
    TimeBinSteps,
)


[docs] def calculate_minimum_toa(da: RawEventProbability) -> MinimumTimeOfArrival: """Calculate the minimum time of arrival from the data.""" return MinimumTimeOfArrival(da.coords['t'].min())
[docs] def calculate_maximum_toa(da: RawEventProbability) -> MaximumTimeOfArrival: """Calculate the maximum time of arrival from the data.""" return MaximumTimeOfArrival(da.coords['t'].max())
[docs] def proton_charge_from_event_counts(da: NMXReducedCounts) -> ProtonCharge: """Make up the proton charge from the event counts. Proton charge is proportional to the number of neutrons, which is proportional to the number of events. The scale factor is manually chosen based on previous results to be convenient for data manipulation in the next steps. It is derived this way since the protons are not part of McStas simulation, and the number of neutrons is not included in the result. Parameters ---------- event_da: The event data """ # Arbitrary number to scale the proton charge return ProtonCharge(sc.scalar(1 / 10_000, unit='dimensionless') * da.data.sum())
[docs] def reduce_raw_event_probability( da: RawEventProbability, pixel_ids: PixelIds, time_bin_step: TimeBinSteps ) -> NMXReducedProbability: return NMXReducedProbability(da.group(pixel_ids).hist(t=time_bin_step))
[docs] def raw_event_probability_to_counts( da: NMXReducedProbability, scale_factor: McStasWeight2CountScaleFactor, ) -> NMXReducedCounts: return NMXReducedCounts(da * scale_factor)
[docs] def format_nmx_reduced_data( da: NMXReducedCounts, proton_charge: ProtonCharge, experiment_metadata: NMXExperimentMetadata, detector_metadata: NMXDetectorMetadata, ) -> NMXReducedDataGroup: """Bin time of arrival data into ``time_bin_step`` bins.""" return NMXReducedDataGroup( sc.DataGroup( counts=da, proton_charge=proton_charge, **experiment_metadata, **detector_metadata, ) )
def _concat_or_same( obj: list[sc.Variable | sc.DataArray], dim: str ) -> sc.Variable | sc.DataArray: first = obj[0] # instrument.to_coords in bin_time_of_arrival adds a panel coord to some fields, # even if it has only length 1. If this is the case we concat, even if identical. # Maybe McStasInstrument.to_coords should be changed to only handle a single # panel, and not perform concatenation? if all(dim not in o.dims and sc.identical(first, o) for o in obj): return first return sc.concat(obj, dim)
[docs] def merge_panels(*panel: NMXReducedDataGroup) -> NMXReducedDataGroup: """Merge a list of panels by concatenating along the 'panel' dimension.""" keys = panel[0].keys() if not all(p.keys() == keys for p in panel): raise ValueError("All panels must have the same keys.") return NMXReducedDataGroup( sc.DataGroup( {key: _concat_or_same([p[key] for p in panel], 'panel') for key in keys} ) )