# SPDX-License-Identifier: BSD-3-Clause# Copyright (c) 2023 Scipp contributors (https://github.com/scipp)"""Components for non-ESS diffraction experiments.WARNING This package will be removed in the future!It only serves as helpers to develop workflows until it is determined,which mechanisms and interfaces will be used at ESS."""frompathlibimportPathimportnumpyasnpimportscippasscimportscippneutronasscnfrom...powder.loggingimportget_loggerdef_as_boolean_mask(var:sc.Variable)->sc.Variable:ifvar.dtypein('float32','float64'):ifsc.any(var!=var.to(dtype='int64')).value:raiseValueError('Cannot construct boolean mask, the input mask has fractional values.')returnvar.to(dtype=bool)def_parse_calibration_instrument_args(filename:str|Path,*,instrument_filename:str|None=None,instrument_name:str|None=None,)->dict[str,str]:ifinstrument_filenameisnotNone:ifinstrument_nameisnotNone:raiseValueError('Only one argument of `instrument_name` and ''`instrument_filename` is allowed, got both.')instrument_arg={'InstrumentFilename':instrument_filename}instrument_message=f'with instrument file {instrument_filename}'else:ifinstrument_nameisNone:raiseValueError('Need one argument of `instrument_name` and ''`instrument_filename` is allowed, got neither.')instrument_arg={'InstrumentName':instrument_name}instrument_message=f'with instrument {instrument_name}'get_logger().info('Loading calibration from file %s%s',filename,instrument_message)returninstrument_arg
[docs]defload_calibration(filename:str|Path,*,instrument_filename:str|None=None,instrument_name:str|None=None,mantid_args:dict|None=None,)->sc.Dataset:""" Load and return calibration data. Warning ------- This function is designed to work with calibration files used by Mantid, specifically for POWGEN. It does not represent the interface used at ESS and will be removed in the future. Uses the Mantid algorithm `LoadDiffCal <https://docs.mantidproject.org/nightly/algorithms/LoadDiffCal-v1.html>`_ and stores the data in a :class:`scipp.Dataset` Note that this function requires mantid to be installed and available in the same Python environment as ess. Parameters ---------- filename: The name of the calibration file to load. instrument_filename: Instrument definition file. instrument_name: Name of the instrument. mantid_args: Dictionary with additional arguments for the `LoadDiffCal` Mantid algorithm. Returns ------- : A Dataset containing the calibration data and masking. """mantid_args={}ifmantid_argsisNoneelsemantid_argsmantid_args.update(_parse_calibration_instrument_args(filename,instrument_filename=instrument_filename,instrument_name=instrument_name,))withscn.mantid.run_mantid_alg('LoadDiffCal',Filename=str(filename),MakeGroupingWorkspace=False,**mantid_args,)asws:ds=scn.from_mantid(ws.OutputCalWorkspace)mask_ws=ws.OutputMaskWorkspacerows=mask_ws.getNumberHistograms()mask=sc.array(dims=['row'],values=np.fromiter((mask_ws.readY(i)[0]foriinrange(rows)),count=rows,dtype=np.bool_),unit=None,)# This is deliberately not stored as a mask since that would make# subsequent handling, e.g., with groupby, more complicated. The mask# is conceptually not masking rows in this table, i.e., it is not# marking invalid rows, but rather describes masking for other data.ds["mask"]=_as_boolean_mask(mask)# The file does not define units# TODO why those units? Can we verify?ds["difc"].unit='us / angstrom'ds["difa"].unit='us / angstrom**2'ds["tzero"].unit='us'ds=ds.rename_dims({'row':'detector'})ds.coords['detector']=ds['detid'].datads.coords['detector'].unit=Nonedelds['detid']returnds