From 5c58a0d439bd54cf959c2f33ac2065e255151b32 Mon Sep 17 00:00:00 2001 From: Roman Tolchenov <roman.tolchenov@stfc.ac.uk> Date: Wed, 16 Jan 2013 14:54:24 +0000 Subject: [PATCH] Re #6131. Implemented changes to version 2 muon nexus format. --- .../Framework/Algorithms/test/AlphaCalcTest.h | 5 +- .../Algorithms/test/ApplyDeadTimeCorrTest.h | 4 +- .../Algorithms/test/MuonAsymmetryCalcTest.h | 4 +- .../Algorithms/test/MuonRemoveExpDecayTest.h | 4 +- .../Algorithms/test/RemoveBinsTest.h | 4 +- .../Framework/DataHandling/CMakeLists.txt | 4 +- .../inc/MantidDataHandling/LoadMuonNexus.h | 20 +- .../inc/MantidDataHandling/LoadMuonNexus1.h | 101 +++ .../inc/MantidDataHandling/LoadMuonNexus2.h | 18 +- .../DataHandling/src/LoadMuonNexus.cpp | 640 +--------------- .../DataHandling/src/LoadMuonNexus1.cpp | 696 ++++++++++++++++++ .../DataHandling/src/LoadMuonNexus2.cpp | 310 ++++---- .../DataHandling/test/GroupDetectors2Test.h | 8 +- .../DataHandling/test/LoadMuonNexus1Test.h | 415 +++++++++++ .../DataHandling/test/LoadMuonNexus2Test.h | 64 +- .../DataHandling/test/LoadMuonNexusTest.h | 381 +--------- .../Nexus/inc/MantidNexus/NexusClasses.h | 7 +- .../Framework/Nexus/src/NexusClasses.cpp | 13 + Test/AutoTestData/deltat_tdc_gpd_0900.nxs | Bin 0 -> 75276 bytes 19 files changed, 1466 insertions(+), 1232 deletions(-) create mode 100644 Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus1.h create mode 100644 Code/Mantid/Framework/DataHandling/src/LoadMuonNexus1.cpp create mode 100644 Code/Mantid/Framework/DataHandling/test/LoadMuonNexus1Test.h create mode 100644 Test/AutoTestData/deltat_tdc_gpd_0900.nxs diff --git a/Code/Mantid/Framework/Algorithms/test/AlphaCalcTest.h b/Code/Mantid/Framework/Algorithms/test/AlphaCalcTest.h index 80b2f9d2ac0..dbaede7f8ca 100644 --- a/Code/Mantid/Framework/Algorithms/test/AlphaCalcTest.h +++ b/Code/Mantid/Framework/Algorithms/test/AlphaCalcTest.h @@ -3,7 +3,7 @@ #include <cxxtest/TestSuite.h> -#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataHandling/LoadMuonNexus2.h" #include "MantidDataHandling/LoadInstrument.h" #include "MantidDataHandling/GroupDetectors.h" #include "MantidAPI/IAlgorithm.h" @@ -38,6 +38,7 @@ public: void testCalAlphaManySpectra() { + //system("pause"); //Load the muon nexus file loader.initialize(); loader.setPropertyValue("Filename", "emu00006473.nxs"); @@ -93,7 +94,7 @@ public: private: AlphaCalc alphaCalc; - Mantid::DataHandling::LoadMuonNexus loader; + Mantid::DataHandling::LoadMuonNexus2 loader; }; diff --git a/Code/Mantid/Framework/Algorithms/test/ApplyDeadTimeCorrTest.h b/Code/Mantid/Framework/Algorithms/test/ApplyDeadTimeCorrTest.h index 230fd5576ff..d729719b217 100644 --- a/Code/Mantid/Framework/Algorithms/test/ApplyDeadTimeCorrTest.h +++ b/Code/Mantid/Framework/Algorithms/test/ApplyDeadTimeCorrTest.h @@ -3,7 +3,7 @@ #include <cxxtest/TestSuite.h> -#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataHandling/LoadMuonNexus2.h" #include "MantidDataHandling/LoadInstrument.h" #include "MantidDataHandling/GroupDetectors.h" #include "MantidAPI/IAlgorithm.h" @@ -199,7 +199,7 @@ public: private: ApplyDeadTimeCorr applyDeadTime; - Mantid::DataHandling::LoadMuonNexus loader; + Mantid::DataHandling::LoadMuonNexus2 loader; }; diff --git a/Code/Mantid/Framework/Algorithms/test/MuonAsymmetryCalcTest.h b/Code/Mantid/Framework/Algorithms/test/MuonAsymmetryCalcTest.h index b6a9857ba81..883b9f5256f 100644 --- a/Code/Mantid/Framework/Algorithms/test/MuonAsymmetryCalcTest.h +++ b/Code/Mantid/Framework/Algorithms/test/MuonAsymmetryCalcTest.h @@ -3,7 +3,7 @@ #include <cxxtest/TestSuite.h> -#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataHandling/LoadMuonNexus2.h" #include "MantidDataHandling/LoadInstrument.h" #include "MantidDataHandling/GroupDetectors.h" #include "MantidAPI/IAlgorithm.h" @@ -91,7 +91,7 @@ public: private: MuonAsymmetryCalc asymCalc; - Mantid::DataHandling::LoadMuonNexus loader; + Mantid::DataHandling::LoadMuonNexus2 loader; Mantid::DataHandling::GroupDetectors group1; Mantid::DataHandling::GroupDetectors group2; diff --git a/Code/Mantid/Framework/Algorithms/test/MuonRemoveExpDecayTest.h b/Code/Mantid/Framework/Algorithms/test/MuonRemoveExpDecayTest.h index a7b6d068e7d..6ad22942904 100644 --- a/Code/Mantid/Framework/Algorithms/test/MuonRemoveExpDecayTest.h +++ b/Code/Mantid/Framework/Algorithms/test/MuonRemoveExpDecayTest.h @@ -4,7 +4,7 @@ #include <cxxtest/TestSuite.h> #include "MantidDataHandling/LoadInstrument.h" -#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataHandling/LoadMuonNexus2.h" #include "MantidAlgorithms/MuonRemoveExpDecay.h" #include "MantidAPI/Workspace.h" #include "MantidDataObjects/Workspace2D.h" @@ -93,7 +93,7 @@ public: private: MuonRemoveExpDecay alg; - Mantid::DataHandling::LoadMuonNexus loader; + Mantid::DataHandling::LoadMuonNexus2 loader; }; diff --git a/Code/Mantid/Framework/Algorithms/test/RemoveBinsTest.h b/Code/Mantid/Framework/Algorithms/test/RemoveBinsTest.h index 545b4331dd9..6e4886bdbcc 100644 --- a/Code/Mantid/Framework/Algorithms/test/RemoveBinsTest.h +++ b/Code/Mantid/Framework/Algorithms/test/RemoveBinsTest.h @@ -9,7 +9,7 @@ #include <stdexcept> #include "MantidAlgorithms/RemoveBins.h" -#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataHandling/LoadMuonNexus2.h" #include "MantidDataHandling/LoadInstrument.h" #include "MantidDataObjects/Workspace2D.h" #include "MantidKernel/UnitFactory.h" @@ -177,7 +177,7 @@ public: void xtestRealData() { - Mantid::DataHandling::LoadMuonNexus loader; + Mantid::DataHandling::LoadMuonNexus2 loader; loader.initialize(); loader.setPropertyValue("Filename", "emu00006473.nxs"); loader.setPropertyValue("OutputWorkspace", "EMU6473"); diff --git a/Code/Mantid/Framework/DataHandling/CMakeLists.txt b/Code/Mantid/Framework/DataHandling/CMakeLists.txt index 9f7787a4cde..74e4d8e267a 100644 --- a/Code/Mantid/Framework/DataHandling/CMakeLists.txt +++ b/Code/Mantid/Framework/DataHandling/CMakeLists.txt @@ -58,6 +58,7 @@ set ( SRC_FILES src/LoadMaskingFile.cpp src/LoadMuonLog.cpp src/LoadMuonNexus.cpp + src/LoadMuonNexus1.cpp src/LoadMuonNexus2.cpp src/LoadNXSPE.cpp src/LoadNexus.cpp @@ -196,6 +197,7 @@ set ( INC_FILES inc/MantidDataHandling/LoadMaskingFile.h inc/MantidDataHandling/LoadMuonLog.h inc/MantidDataHandling/LoadMuonNexus.h + inc/MantidDataHandling/LoadMuonNexus1.h inc/MantidDataHandling/LoadMuonNexus2.h inc/MantidDataHandling/LoadNXSPE.h inc/MantidDataHandling/LoadNexus.h @@ -322,8 +324,8 @@ set ( TEST_FILES LoadMappingTableTest.h LoadMaskTest.h LoadMuonLogTest.h + LoadMuonNexus1Test.h LoadMuonNexus2Test.h - LoadMuonNexusTest.h LoadNXSPETest.h LoadNexusLogsTest.h LoadNexusMonitorsTest.h diff --git a/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus.h b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus.h index 579b210cb9f..74ba34c15d3 100644 --- a/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus.h +++ b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus.h @@ -22,10 +22,8 @@ namespace Mantid { /** @class LoadMuonNexus LoadMuonNexus.h DataHandling/LoadMuonNexus.h - Loads an file in Nexus Muon format and stores it in a 2D workspace - (Workspace2D class). LoadMuonNexus is an algorithm and as such inherits - from the Algorithm class, via DataHandlingCommand, and overrides - the init() & exec() methods. + It is a base class for loadres for versions 1 and 2 of the muon nexus file format. + It implements property initialization and some common for both versions methods. Required Properties: <UL> @@ -77,19 +75,20 @@ namespace Mantid /// Algorithm's category for identification overriding a virtual method virtual const std::string category() const { return "DataHandling\\Nexus;Muon"; } - /// do a quick check that this file can be loaded + /// do a quick check that this file can be loaded virtual bool quickFileCheck(const std::string& filePath,size_t nread,const file_header& header); /// check the structure of the file and return a value between 0 and 100 of how much this file can be loaded virtual int fileCheck(const std::string& filePath); protected: - /// Overwrites Algorithm method - void exec(); + virtual void runLoadInstrumentFromNexus(DataObjects::Workspace2D_sptr) {} void checkOptionalProperties(); void runLoadInstrument(DataObjects::Workspace2D_sptr); /// The name and path of the input file std::string m_filename; + /// The first top-level entry name in the file + std::string m_entry_name; /// The instrument name from Nexus std::string m_instrument_name; /// The sample name read from Nexus @@ -120,13 +119,6 @@ namespace Mantid virtual void initDocs(); /// Overwrites Algorithm method. void init(); - - void loadData(const MantidVecPtr::ptr_type& tcbs,size_t hist, specid_t& i, - MuonNexusReader& nxload, const int64_t lengthIn, DataObjects::Workspace2D_sptr localWorkspace); - void runLoadInstrumentFromNexus(DataObjects::Workspace2D_sptr); - void runLoadMappingTable(DataObjects::Workspace2D_sptr); - void runLoadLog(DataObjects::Workspace2D_sptr); - void loadRunDetails(DataObjects::Workspace2D_sptr localWorkspace); }; } // namespace DataHandling diff --git a/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus1.h b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus1.h new file mode 100644 index 00000000000..671b0315371 --- /dev/null +++ b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus1.h @@ -0,0 +1,101 @@ +#ifndef MANTID_DATAHANDLING_LOADMUONNEXUS1_H_ +#define MANTID_DATAHANDLING_LOADMUONNEXUS1_H_ + +//---------------------------------------------------------------------- +// Includes +//---------------------------------------------------------------------- +#include "MantidDataHandling/LoadMuonNexus.h" +#include "MantidDataObjects/Workspace2D.h" +#include "MantidKernel/System.h" +#include "MantidAPI/SpectraDetectorMap.h" + +#include <napi.h> +//---------------------------------------------------------------------- +// Forward declaration +//---------------------------------------------------------------------- +class MuonNexusReader; + +namespace Mantid +{ + namespace DataHandling + { + /** @class LoadMuonNexus LoadMuonNexus.h DataHandling/LoadMuonNexus.h + + Loads an file in Nexus Muon format version 1 and stores it in a 2D workspace + (Workspace2D class). LoadMuonNexus is an algorithm and as such inherits + from the Algorithm class, via DataHandlingCommand, and overrides + the init() & exec() methods. + + Required Properties: + <UL> + <LI> Filename - The name of and path to the input NeXus file </LI> + <LI> OutputWorkspace - The name of the workspace in which to store the imported data + (a multiperiod file will store higher periods in workspaces called OutputWorkspace_PeriodNo) + [ not yet implemented for Nexus ]</LI> + </UL> + + Optional Properties: (note that these options are not available if reading a multiperiod file) + <UL> + <LI> spectrum_min - The spectrum to start loading from</LI> + <LI> spectrum_max - The spectrum to load to</LI> + <LI> spectrum_list - An ArrayProperty of spectra to load</LI> + <LI> auto_group - Determines whether the spectra are automatically grouped together based on the groupings in the NeXus file. </LI> + </UL> + + Copyright © 2007-2010 ISIS Rutherford Appleton Laboratory & NScD Oak Ridge National Laboratory + + This file is part of Mantid. + + Mantid is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 3 of the License, or + (at your option) any later version. + + Mantid is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see <http://www.gnu.org/licenses/>. + + File change history is stored at: <https://github.com/mantidproject/mantid>. + Code Documentation is available at: <http://doxygen.mantidproject.org> + */ + class DLLExport LoadMuonNexus1 : public LoadMuonNexus + { + public: + /// Default constructor + LoadMuonNexus1(); + /// Destructor + virtual ~LoadMuonNexus1() {} + /// Algorithm's name for identification overriding a virtual method + virtual const std::string name() const { return "LoadMuonNexus"; } + /// Algorithm's version for identification overriding a virtual method + virtual int version() const { return 1; } + /// Algorithm's category for identification overriding a virtual method + virtual const std::string category() const { return "DataHandling\\Nexus;Muon"; } + + /// check the structure of the file and return a value between 0 and 100 of how much this file can be loaded + virtual int fileCheck(const std::string& filePath); + protected: + /// Overwrites Algorithm method + void exec(); + /// Implement the base class method + void runLoadInstrumentFromNexus(DataObjects::Workspace2D_sptr); + + private: + /// Sets documentation strings for this algorithm + virtual void initDocs(); + + void loadData(const MantidVecPtr::ptr_type& tcbs,size_t hist, specid_t& i, + MuonNexusReader& nxload, const int64_t lengthIn, DataObjects::Workspace2D_sptr localWorkspace); + void runLoadMappingTable(DataObjects::Workspace2D_sptr); + void runLoadLog(DataObjects::Workspace2D_sptr); + void loadRunDetails(DataObjects::Workspace2D_sptr localWorkspace); + }; + + } // namespace DataHandling +} // namespace Mantid + +#endif /*MANTID_DATAHANDLING_LOADMUONNEXUS1_H_*/ diff --git a/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus2.h b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus2.h index edd5b5ed44c..1094ac22d00 100644 --- a/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus2.h +++ b/Code/Mantid/Framework/DataHandling/inc/MantidDataHandling/LoadMuonNexus2.h @@ -10,24 +10,12 @@ namespace Mantid { -namespace NeXus -{ - // Forward declarations - template<class T> - class NXDataSetTyped; - typedef NXDataSetTyped<int> NXInt; -} - namespace DataHandling { - // Forward declarations - template<class T> - class NXDataSetTyped; - typedef NXDataSetTyped<int> NXInt; /** - Loads an file in NeXus Muon format and stores it in a 2D workspace + Loads an file in NeXus Muon format version 1 and 2 and stores it in a 2D workspace (Workspace2D class). LoadMuonNexus is an algorithm and as such inherits from the Algorithm class, via DataHandlingCommand, and overrides the init() & exec() methods. @@ -82,8 +70,6 @@ namespace NeXus /// Algorithm's category for identification overriding a virtual method virtual const std::string category() const { return "DataHandling\\Nexus;Muon"; } - /// do a quick check that this file can be loaded - virtual bool quickFileCheck(const std::string& filePath,size_t nread,const file_header& header); /// check the structure of the file and return a value between 0 and 100 of how much this file can be loaded virtual int fileCheck(const std::string& filePath); private: @@ -91,6 +77,8 @@ namespace NeXus virtual void initDocs(); /// Overwrites Algorithm method void exec(); + /// Execute this version of the algorithm + void doExec(); void loadData(const Mantid::NeXus::NXInt& counts,const std::vector<double>& timeBins,int wsIndex, int period,int spec,API::MatrixWorkspace_sptr localWorkspace); diff --git a/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus.cpp b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus.cpp index 53ea3c2f104..e043b3f973e 100644 --- a/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus.cpp +++ b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus.cpp @@ -1,45 +1,3 @@ -/*WIKI* - - -The algorithm LoadMuonNexus will read a Muon Nexus data file (original format) and place the data -into the named workspace. -The file name can be an absolute or relative path and should have the extension -.nxs or .NXS. -If the file contains data for more than one period, a separate workspace will be generated for each. -After the first period the workspace names will have "_2", "_3", and so on, appended to the given workspace name. -For single period data, the optional parameters can be used to control which spectra are loaded into the workspace. -If spectrum_min and spectrum_max are given, then only that range to data will be loaded. -If a spectrum_list is given than those values will be loaded. -* TODO get XML descriptions of Muon instruments. This data is not in existing Muon Nexus files. -* TODO load the spectra detector mapping. This may be very simple for Muon instruments. - -===Time series data=== -The log data in the Nexus file (NX_LOG sections) will be loaded as TimeSeriesProperty data within the workspace. -Time is stored as seconds from the Unix epoch. - -===Errors=== - -The error for each histogram count is set as the square root of the number of counts. - -===Time bin data=== - -The ''corrected_times'' field of the Nexus file is used to provide time bin data and the bin edge values are calculated from these -bin centre times. - -===Multiperiod data=== - -To determine if a file contains data from more than one period the field ''switching_states'' is read from the Nexus file. -If this value is greater than one it is taken to be the number of periods, <math>N_p</math> of the data. -In this case the <math>N_s</math> spectra in the ''histogram_data'' field are split with <math>N_s/N_p</math> assigned to each period. - -===ChildAlgorithms used=== - -The ChildAlgorithms used by LoadMuonNexus are: -* LoadMuonLog - this reads log information from the Nexus file and uses it to create TimeSeriesProperty entries in the workspace. -* LoadInstrument - this algorithm looks for an XML description of the instrument and if found reads it. -* LoadIntstrumentFromNexus - this is called if the normal LoadInstrument fails. As the Nexus file has limited instrument data, this only populates a few fields. - -*WIKI*/ //---------------------------------------------------------------------- // Includes //---------------------------------------------------------------------- @@ -70,8 +28,6 @@ namespace Mantid { namespace DataHandling { - // Register the algorithm into the algorithm factory - DECLARE_ALGORITHM(LoadMuonNexus) /// Sets documentation strings for this algorithm void LoadMuonNexus::initDocs() @@ -139,396 +95,6 @@ namespace Mantid "The name of the vector in which to store the list of deadtimes for each spectrum", Direction::Output); } - /** Executes the algorithm. Reading in the file and creating and populating - * the output workspace - * - * @throw Exception::FileError If the Nexus file cannot be found/opened - * @throw std::invalid_argument If the optional properties are set to invalid values - */ - void LoadMuonNexus::exec() - { - // Retrieve the filename from the properties - m_filename = getPropertyValue("Filename"); - // Retrieve the entry number - m_entrynumber = getProperty("EntryNumber"); - - - - NXRoot root(m_filename); - NXEntry entry = root.openEntry("run/histogram_data_1"); - try - { - NXInfo info = entry.getDataSetInfo("time_zero"); - if (info.stat != NX_ERROR) - { - double dum = root.getFloat("run/histogram_data_1/time_zero"); - setProperty("TimeZero", dum); - } - } - catch (::NeXus::Exception&) - {} - - try - { - NXInfo infoResolution = entry.getDataSetInfo("resolution"); - NXInt counts = root.openNXInt("run/histogram_data_1/counts"); - std::string firstGoodBin = counts.attributes("first_good_bin"); - if ( !firstGoodBin.empty() && infoResolution.stat != NX_ERROR ) - { - double bin = static_cast<double>(boost::lexical_cast<int>(firstGoodBin)); - double bin_size = static_cast<double>(root.getInt("run/histogram_data_1/resolution"))/1000000.0; - setProperty("FirstGoodData", bin*bin_size); - } - } - catch (::NeXus::Exception&) - {} - - try - { - std::vector<double>defaultDeadTimes; - NXFloat deadTimes = root.openNXFloat("run/instrument/detector/deadtimes"); - deadTimes.load(); - - int length = deadTimes.dim0(); - for (int i = 0; i < length; i++) - { - defaultDeadTimes.push_back(static_cast<double>(*(deadTimes() + i) ) ); - } - setProperty("DeadTimes", defaultDeadTimes); - } - catch (::NeXus::Exception&) - {} - - NXEntry nxRun = root.openEntry("run"); - std::string title; - std::string notes; - try - { - title = nxRun.getString("title"); - notes = nxRun.getString("notes"); - } - catch (::NeXus::Exception&) - {} - std::string run_num; - try - { - run_num = boost::lexical_cast<std::string>(nxRun.getInt("number")); - } - catch (::NeXus::Exception&) - {} - - MuonNexusReader nxload; - if (nxload.readFromFile(m_filename) != 0) - { - g_log.error("Unable to open file " + m_filename); - throw Exception::FileError("Unable to open File:" , m_filename); - } - - // Read in the instrument name from the Nexus file - m_instrument_name = nxload.getInstrumentName(); - // Read in the number of spectra in the Nexus file - m_numberOfSpectra = nxload.t_nsp1; - if(m_entrynumber!=0) - { - m_numberOfPeriods=1; - if(m_entrynumber>nxload.t_nper) - { - throw std::invalid_argument("Invalid Entry Number:Enter a valid number"); - } - } - else - { - // Read the number of periods in this file - m_numberOfPeriods = nxload.t_nper; - } - // Need to extract the user-defined output workspace name - Property *ws = getProperty("OutputWorkspace"); - std::string localWSName = ws->value(); - // If multiperiod, will need to hold the Instrument, Sample & SpectraDetectorMap for copying - boost::shared_ptr<Instrument> instrument; - boost::shared_ptr<Sample> sample; - - // Call private method to validate the optional parameters, if set - checkOptionalProperties(); - - // Read the number of time channels (i.e. bins) from the Nexus file - const int channelsPerSpectrum = nxload.t_ntc1; - // Read in the time bin boundaries - const int lengthIn = channelsPerSpectrum + 1; - float* timeChannels = new float[lengthIn]; - nxload.getTimeChannels(timeChannels, lengthIn); - // Put the read in array into a vector (inside a shared pointer) - boost::shared_ptr<MantidVec> timeChannelsVec - (new MantidVec(timeChannels, timeChannels + lengthIn)); - - // Calculate the size of a workspace, given its number of periods & spectra to read - int64_t total_specs; - if( m_interval || m_list) - { - total_specs = m_spec_list.size(); - if (m_interval) - { - total_specs += (m_spec_max-m_spec_min+1); - m_spec_max += 1; - } - } - else - { - total_specs = m_numberOfSpectra; - // for nexus return all spectra - m_spec_min = 0; // changed to 0 for NeXus, was 1 for Raw - m_spec_max = m_numberOfSpectra; // was +1? - } - - // Create the 2D workspace for the output - DataObjects::Workspace2D_sptr localWorkspace = boost::dynamic_pointer_cast<DataObjects::Workspace2D> - (WorkspaceFactory::Instance().create("Workspace2D",total_specs,lengthIn,lengthIn-1)); - localWorkspace->setTitle(title); - localWorkspace->setComment(notes); - localWorkspace->mutableRun().addLogData(new PropertyWithValue<std::string>("run_number", run_num)); - // Set the unit on the workspace to muon time, for now in the form of a Label Unit - boost::shared_ptr<Kernel::Units::Label> lblUnit = - boost::dynamic_pointer_cast<Kernel::Units::Label>(UnitFactory::Instance().create("Label")); - lblUnit->setLabel("Time","microsecond"); - localWorkspace->getAxis(0)->unit() = lblUnit; - // Set y axis unit - localWorkspace->setYUnit("Counts"); - - WorkspaceGroup_sptr wsGrpSptr=WorkspaceGroup_sptr(new WorkspaceGroup); - if(m_numberOfPeriods>1) - { - setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(wsGrpSptr)); - } - - API::Progress progress(this,0.,1.,m_numberOfPeriods * total_specs); - // Loop over the number of periods in the Nexus file, putting each period in a separate workspace - for (int64_t period = 0; period < m_numberOfPeriods; ++period) { - if(m_entrynumber!=0) - { - period=m_entrynumber-1; - if(period!=0) - { - loadRunDetails(localWorkspace); - runLoadInstrument(localWorkspace ); - runLoadMappingTable(localWorkspace ); - } - } - - if (period == 0) - { - // Only run the Child Algorithms once - loadRunDetails(localWorkspace); - runLoadInstrument(localWorkspace ); - runLoadMappingTable(localWorkspace ); - runLoadLog(localWorkspace ); - localWorkspace->populateInstrumentParameters(); - } - else // We are working on a higher period of a multiperiod raw file - { - localWorkspace = boost::dynamic_pointer_cast<DataObjects::Workspace2D> - (WorkspaceFactory::Instance().create(localWorkspace)); - localWorkspace->setTitle(title); - localWorkspace->setComment(notes); - //localWorkspace->newInstrumentParameters(); ??? - - } - - - std::string outws(""); - if(m_numberOfPeriods>1) - { - std::string outputWorkspace = "OutputWorkspace"; - std::stringstream suffix; - suffix << (period+1); - outws =outputWorkspace+"_"+suffix.str(); - std::string WSName = localWSName + "_" + suffix.str(); - declareProperty(new WorkspaceProperty<Workspace>(outws,WSName,Direction::Output)); - if (wsGrpSptr) - { - wsGrpSptr->addWorkspace( localWorkspace ); - } - } - - size_t counter = 0; - for (int64_t i = m_spec_min; i < m_spec_max; ++i) - { - // Shift the histogram to read if we're not in the first period - specid_t histToRead = static_cast<specid_t>(i + period*total_specs); - loadData(timeChannelsVec,counter,histToRead,nxload,lengthIn-1,localWorkspace ); // added -1 for NeXus - counter++; - progress.report(); - } - // Read in the spectra in the optional list parameter, if set - if (m_list) - { - for(size_t i=0; i < m_spec_list.size(); ++i) - { - loadData(timeChannelsVec,counter,m_spec_list[i],nxload,lengthIn-1, localWorkspace ); - counter++; - progress.report(); - } - } - // Just a sanity check - assert(counter == size_t(total_specs) ); - - bool autogroup = getProperty("AutoGroup"); - - if (autogroup) - { - - //Get the groupings - int64_t max_group = 0; - // use a map for mapping group number and output workspace index in case - // there are group numbers > number of groups - std::map<int64_t,int64_t> groups; - m_groupings.resize(nxload.numDetectors); - bool thereAreZeroes = false; - for (int64_t i =0; i < static_cast<int64_t>(nxload.numDetectors); ++i) - { - int64_t ig = static_cast<int64_t>(nxload.detectorGroupings[i]); - if (ig == 0) - { - thereAreZeroes = true; - continue; - } - m_groupings[i] = static_cast<specid_t>(ig); - if (groups.find(ig) == groups.end()) - groups[ig] = static_cast<int64_t>(groups.size()); - if (ig > max_group) max_group = ig; - } - - if (thereAreZeroes) - for (int64_t i =0; i < static_cast<int64_t>(nxload.numDetectors); ++i) - { - int64_t ig = static_cast<int64_t>(nxload.detectorGroupings[i]); - if (ig == 0) - { - ig = ++max_group; - m_groupings[i] = static_cast<specid_t>(ig); - groups[ig] = groups.size(); - } - } - - int numHists = static_cast<int>(localWorkspace->getNumberHistograms()); - size_t ngroups = groups.size(); // number of groups - - // to output groups in ascending order - { - int64_t i=0; - for(std::map<int64_t,int64_t>::iterator it=groups.begin();it!=groups.end();++it,++i) - { - it->second = i; - g_log.information()<<"group "<<it->first<<": "; - bool first = true; - int64_t first_i = -1 * std::numeric_limits<int64_t>::max(); - int64_t last_i = -1 * std::numeric_limits<int64_t>::max(); - for(int64_t i=0;i<static_cast<int64_t>(numHists);i++) - if (m_groupings[i] == it->first) - { - if (first) - { - first = false; - g_log.information()<<i; - first_i = i; - } - else - { - if (first_i >= 0) - { - if (i > last_i + 1) - { - g_log.information()<<'-'<<i; - first_i = -1; - } - } - else - { - g_log.information()<<','<<i; - first_i = i; - } - } - last_i = i; - } - else - { - if (!first && first_i >= 0) - { - if (last_i > first_i) - g_log.information()<<'-'<<last_i; - first_i = -1; - } - } - if (first_i >= 0 && last_i > first_i) - g_log.information()<<'-'<<last_i; - g_log.information()<<'\n'; - } - } - - //Create a workspace with only two spectra for forward and back - DataObjects::Workspace2D_sptr groupedWS = boost::dynamic_pointer_cast<DataObjects::Workspace2D> - (API::WorkspaceFactory::Instance().create(localWorkspace, ngroups, localWorkspace->dataX(0).size(), localWorkspace->blocksize())); - - boost::shared_array<specid_t> spec(new specid_t[numHists]); - boost::shared_array<detid_t> dets(new detid_t[numHists]); - - //Compile the groups - for (int i = 0; i < numHists; ++i) - { - specid_t k = static_cast<specid_t>(groups[ m_groupings[numHists*period + i] ]); - - for (detid_t j = 0; j < static_cast<detid_t>(localWorkspace->blocksize()); ++j) - { - groupedWS->dataY(k)[j] = groupedWS->dataY(k)[j] + localWorkspace->dataY(i)[j]; - - //Add the errors in quadrature - groupedWS->dataE(k)[j] - = sqrt(pow(groupedWS->dataE(k)[j], 2) + pow(localWorkspace->dataE(i)[j], 2)); - } - - //Copy all the X data - groupedWS->dataX(k) = localWorkspace->dataX(i); - spec[i] = k + 1; - dets[i] = i + 1; - } - - m_groupings.clear(); - - // All two spectra - for(detid_t k=0; k<static_cast<detid_t>(ngroups); k++) - { - groupedWS->getAxis(1)->spectraNo(k)= k + 1; - } - - groupedWS->replaceSpectraMap(new API::SpectraDetectorMap(spec.get(),dets.get(),numHists)); - - // Assign the result to the output workspace property - if(m_numberOfPeriods>1) - setProperty(outws, boost::dynamic_pointer_cast<Workspace>(groupedWS)); - else - { - setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(groupedWS)); - - } - - } - else - { - // Assign the result to the output workspace property - if(m_numberOfPeriods>1) - setProperty(outws,boost::dynamic_pointer_cast<Workspace>(localWorkspace)); - else - { - setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(localWorkspace)); - } - - } - - } // loop over periods - - // Clean up - delete[] timeChannels; - } - /// Validates the optional 'spectra to read' properties, if they have been set void LoadMuonNexus::checkOptionalProperties() { @@ -564,91 +130,6 @@ namespace Mantid } } - /** Load in a single spectrum taken from a NeXus file - * @param tcbs :: The vector containing the time bin boundaries - * @param hist :: The workspace index - * @param i :: The spectrum number - * @param nxload :: A reference to the MuonNeXusReader object - * @param lengthIn :: The number of elements in a spectrum - * @param localWorkspace :: A pointer to the workspace in which the data will be stored - */ - void LoadMuonNexus::loadData(const MantidVecPtr::ptr_type& tcbs,size_t hist, specid_t& i, - MuonNexusReader& nxload, const int64_t lengthIn, DataObjects::Workspace2D_sptr localWorkspace) - { - // Read in a spectrum - // Put it into a vector, discarding the 1st entry, which is rubbish - // But note that the last (overflow) bin is kept - // For Nexus, not sure if above is the case, hence give all data for now - MantidVec& Y = localWorkspace->dataY(hist); - Y.assign(nxload.counts + i * lengthIn, nxload.counts + i * lengthIn + lengthIn); - // Create and fill another vector for the errors, containing sqrt(count) - MantidVec& E = localWorkspace->dataE(hist); - typedef double (*uf)(double); - uf dblSqrt = std::sqrt; - std::transform(Y.begin(), Y.end(), E.begin(), dblSqrt); - // Populate the workspace. Loop starts from 1, hence i-1 - localWorkspace->setX(hist, tcbs); - localWorkspace->getAxis(1)->spectraNo(hist)= static_cast<int>(hist) + 1; - } - - - /** Log the run details from the file - * @param localWorkspace :: The workspace details to use - */ - void LoadMuonNexus::loadRunDetails(DataObjects::Workspace2D_sptr localWorkspace) - { - API::Run & runDetails = localWorkspace->mutableRun(); - - runDetails.addProperty("run_title", localWorkspace->getTitle(), true); - - int numSpectra = static_cast<int>(localWorkspace->getNumberHistograms()); - runDetails.addProperty("nspectra", numSpectra); - - NXRoot root(m_filename); - try - { - std::string start_time = root.getString("run/start_time"); - runDetails.addProperty("run_start", start_time); - } - catch (std::runtime_error &) - { - g_log.warning("run/start_time is not available, run_start log not added."); - } - - - try - { - std::string stop_time = root.getString("run/stop_time"); - runDetails.addProperty("run_end", stop_time); - } - catch (std::runtime_error &) - { - g_log.warning("run/stop_time is not available, run_end log not added."); - } - - try - { - std::string dur = root.getString("run/duration"); - runDetails.addProperty("dur", dur); - runDetails.addProperty("durunits", 1); // 1 means second here - runDetails.addProperty("dur_secs", dur); - } - catch (std::runtime_error &) - { - g_log.warning("run/duration is not available, dur log not added."); - } - - // Get number of good frames - NXEntry runInstrumentBeam = root.openEntry("run/instrument/beam"); - NXInfo infoGoodTotalFrames = runInstrumentBeam.getDataSetInfo("frames_good"); - if (infoGoodTotalFrames.stat != NX_ERROR) - { - int dum = root.getInt("run/instrument/beam/frames_good"); - runDetails.addProperty("goodfrm", dum); - } - - } - /// Run the Child Algorithm LoadInstrument (or LoadInstrumentFromNexus) void LoadMuonNexus::runLoadInstrument(DataObjects::Workspace2D_sptr localWorkspace) { @@ -682,104 +163,6 @@ namespace Mantid } } - /// Run LoadInstrumentFromNexus as a Child Algorithm (only if loading from instrument definition file fails) - void LoadMuonNexus::runLoadInstrumentFromNexus(DataObjects::Workspace2D_sptr localWorkspace) - { - g_log.information() << "Instrument definition file not found. Attempt to load information about \n" - << "the instrument from nexus data file.\n"; - - IAlgorithm_sptr loadInst = createChildAlgorithm("LoadInstrumentFromNexus"); - - // Now execute the Child Algorithm. Catch and log any error, but don't stop. - bool executionSuccessful(true); - try - { - loadInst->setPropertyValue("Filename", m_filename); - loadInst->setProperty<MatrixWorkspace_sptr> ("Workspace", localWorkspace); - loadInst->execute(); - } - catch( std::invalid_argument&) - { - g_log.information("Invalid argument to LoadInstrument Child Algorithm"); - executionSuccessful = false; - } - catch (std::runtime_error&) - { - g_log.information("Unable to successfully run LoadInstrument Child Algorithm"); - executionSuccessful = false; - } - - if ( !executionSuccessful ) g_log.error("No instrument definition loaded"); - } - - /// Run the LoadMappingTable Child Algorithm to fill the SpectraToDetectorMap - void LoadMuonNexus::runLoadMappingTable(DataObjects::Workspace2D_sptr localWorkspace) - { - NXRoot root(m_filename); - NXInt number = root.openNXInt("run/instrument/detector/number"); - number.load(); - detid_t ndet = static_cast<detid_t>(number[0]/m_numberOfPeriods); - boost::shared_array<detid_t> det(new detid_t[ndet]); - for(detid_t i=0;i<ndet;i++) - { - det[i] = i + 1; - } - localWorkspace->replaceSpectraMap(new API::SpectraDetectorMap(det.get(),det.get(),ndet)); - } - - /// Run the LoadLog Child Algorithm - void LoadMuonNexus::runLoadLog(DataObjects::Workspace2D_sptr localWorkspace) - { - IAlgorithm_sptr loadLog = createChildAlgorithm("LoadMuonLog"); - // Pass through the same input filename - loadLog->setPropertyValue("Filename",m_filename); - // Set the workspace property to be the same one filled above - loadLog->setProperty<MatrixWorkspace_sptr>("Workspace",localWorkspace); - - // Now execute the Child Algorithm. Catch and log any error, but don't stop. - try - { - loadLog->execute(); - } - catch (std::runtime_error&) - { - g_log.error("Unable to successfully run LoadLog Child Algorithm"); - } - catch (std::logic_error&) - { - g_log.error("Unable to successfully run LoadLog Child Algorithm"); - } - - if ( ! loadLog->isExecuted() ) g_log.error("Unable to successfully run LoadLog Child Algorithm"); - - - NXRoot root(m_filename); - - try - { - NXChar orientation = root.openNXChar("run/instrument/detector/orientation"); - // some files have no data there - orientation.load(); - - if (orientation[0] == 't') - { - Kernel::TimeSeriesProperty<double>* p = new Kernel::TimeSeriesProperty<double>("fromNexus"); - std::string start_time = root.getString("run/start_time"); - p->addValue(start_time,-90.0); - localWorkspace->mutableRun().addLogData(p); - setProperty("MainFieldDirection", "Transverse"); - } - else - { - setProperty("MainFieldDirection", "Longitudinal"); - } - } - catch(...) - { - setProperty("MainFieldDirection", "Longitudinal"); - } - - } /**This method does a quick file type check by looking at the first 100 bytes of the file * @param filePath- path of the file including name. @@ -807,7 +190,6 @@ namespace Mantid return true; } return false; - } /**checks the file by opening it and reading few lines * @param filePath :: name of the file inluding its path @@ -815,26 +197,8 @@ namespace Mantid */ int LoadMuonNexus::fileCheck(const std::string& filePath) { - using namespace ::NeXus; - int confidence(0); - try - { - ::NeXus::File file = ::NeXus::File(filePath); - file.openPath("/run/analysis"); - std::string analysisType = file.getStrData(); - std::string compareString = "muon"; - if( analysisType.compare(0,compareString.length(),compareString) == 0 ) - { - // If all this succeeded then we'll assume this is an ISIS Muon NeXus file - confidence = 80; - } - file.close(); - } - catch(::NeXus::Exception&) - { - } - return confidence; - + UNUSED_ARG( filePath ); + throw Kernel::Exception::NotImplementedError("LoadMuonNexus cannot be used as a file checker."); } } // namespace DataHandling diff --git a/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus1.cpp b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus1.cpp new file mode 100644 index 00000000000..ce959c6405d --- /dev/null +++ b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus1.cpp @@ -0,0 +1,696 @@ +/*WIKI* + + +The algorithm LoadMuonNexus will read a Muon Nexus data file (original format) and place the data +into the named workspace. +The file name can be an absolute or relative path and should have the extension +.nxs or .NXS. +If the file contains data for more than one period, a separate workspace will be generated for each. +After the first period the workspace names will have "_2", "_3", and so on, appended to the given workspace name. +For single period data, the optional parameters can be used to control which spectra are loaded into the workspace. +If spectrum_min and spectrum_max are given, then only that range to data will be loaded. +If a spectrum_list is given than those values will be loaded. +* TODO get XML descriptions of Muon instruments. This data is not in existing Muon Nexus files. +* TODO load the spectra detector mapping. This may be very simple for Muon instruments. + +===Time series data=== +The log data in the Nexus file (NX_LOG sections) will be loaded as TimeSeriesProperty data within the workspace. +Time is stored as seconds from the Unix epoch. + +===Errors=== + +The error for each histogram count is set as the square root of the number of counts. + +===Time bin data=== + +The ''corrected_times'' field of the Nexus file is used to provide time bin data and the bin edge values are calculated from these +bin centre times. + +===Multiperiod data=== + +To determine if a file contains data from more than one period the field ''switching_states'' is read from the Nexus file. +If this value is greater than one it is taken to be the number of periods, <math>N_p</math> of the data. +In this case the <math>N_s</math> spectra in the ''histogram_data'' field are split with <math>N_s/N_p</math> assigned to each period. + +===ChildAlgorithms used=== + +The ChildAlgorithms used by LoadMuonNexus are: +* LoadMuonLog - this reads log information from the Nexus file and uses it to create TimeSeriesProperty entries in the workspace. +* LoadInstrument - this algorithm looks for an XML description of the instrument and if found reads it. +* LoadIntstrumentFromNexus - this is called if the normal LoadInstrument fails. As the Nexus file has limited instrument data, this only populates a few fields. + +*WIKI*/ +//---------------------------------------------------------------------- +// Includes +//---------------------------------------------------------------------- +#include "MantidDataHandling/LoadMuonNexus1.h" +#include "MantidDataObjects/Workspace2D.h" +#include "MantidAPI/LoadAlgorithmFactory.h" +#include "MantidAPI/FileProperty.h" +#include "MantidAPI/Progress.h" +#include "MantidAPI/SpectraDetectorMap.h" +#include "MantidAPI/TableRow.h" +#include "MantidGeometry/Instrument/Detector.h" +#include "MantidKernel/UnitFactory.h" +#include "MantidKernel/ConfigService.h" +#include "MantidKernel/ArrayProperty.h" +#include "MantidKernel/TimeSeriesProperty.h" +#include "MantidKernel/BoundedValidator.h" +#include "MantidKernel/ListValidator.h" +#include "MantidNexus/MuonNexusReader.h" +#include "MantidNexus/NexusClasses.h" + +#include <Poco/Path.h> +#include <limits> +#include <cmath> +#include <boost/shared_ptr.hpp> + +namespace Mantid +{ + namespace DataHandling + { + // Register the algorithm into the algorithm factory + DECLARE_ALGORITHM(LoadMuonNexus1) + DECLARE_LOADALGORITHM(LoadMuonNexus1) + + /// Sets documentation strings for this algorithm + void LoadMuonNexus1::initDocs() + { + this->setWikiSummary("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 1 and use the results to populate the named workspace. LoadMuonNexus may be invoked by [[LoadNexus]] if it is given a NeXus file of this type. "); + this->setOptionalMessage("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 1 and use the results to populate the named workspace. LoadMuonNexus may be invoked by LoadNexus if it is given a NeXus file of this type."); + } + + + using namespace Kernel; + using namespace API; + using Geometry::Instrument; + using namespace Mantid::NeXus; + + /// Empty default constructor + LoadMuonNexus1::LoadMuonNexus1() : LoadMuonNexus() {} + + /** Executes the algorithm. Reading in the file and creating and populating + * the output workspace + * + * @throw Exception::FileError If the Nexus file cannot be found/opened + * @throw std::invalid_argument If the optional properties are set to invalid values + */ + void LoadMuonNexus1::exec() + { + // Retrieve the filename from the properties + m_filename = getPropertyValue("Filename"); + // Retrieve the entry number + m_entrynumber = getProperty("EntryNumber"); + + NXRoot root(m_filename); + NXEntry entry = root.openEntry("run/histogram_data_1"); + try + { + NXInfo info = entry.getDataSetInfo("time_zero"); + if (info.stat != NX_ERROR) + { + double dum = root.getFloat("run/histogram_data_1/time_zero"); + setProperty("TimeZero", dum); + } + } + catch (::NeXus::Exception&) + {} + + try + { + NXInfo infoResolution = entry.getDataSetInfo("resolution"); + NXInt counts = root.openNXInt("run/histogram_data_1/counts"); + std::string firstGoodBin = counts.attributes("first_good_bin"); + if ( !firstGoodBin.empty() && infoResolution.stat != NX_ERROR ) + { + double bin = static_cast<double>(boost::lexical_cast<int>(firstGoodBin)); + double bin_size = static_cast<double>(root.getInt("run/histogram_data_1/resolution"))/1000000.0; + setProperty("FirstGoodData", bin*bin_size); + } + } + catch (::NeXus::Exception&) + {} + + try + { + std::vector<double>defaultDeadTimes; + NXFloat deadTimes = root.openNXFloat("run/instrument/detector/deadtimes"); + deadTimes.load(); + + int length = deadTimes.dim0(); + for (int i = 0; i < length; i++) + { + defaultDeadTimes.push_back(static_cast<double>(*(deadTimes() + i) ) ); + } + setProperty("DeadTimes", defaultDeadTimes); + } + catch (::NeXus::Exception&) + {} + + NXEntry nxRun = root.openEntry("run"); + std::string title; + std::string notes; + try + { + title = nxRun.getString("title"); + notes = nxRun.getString("notes"); + } + catch (::NeXus::Exception&) + {} + std::string run_num; + try + { + run_num = boost::lexical_cast<std::string>(nxRun.getInt("number")); + } + catch (::NeXus::Exception&) + {} + + MuonNexusReader nxload; + if (nxload.readFromFile(m_filename) != 0) + { + g_log.error("Unable to open file " + m_filename); + throw Exception::FileError("Unable to open File:" , m_filename); + } + + // Read in the instrument name from the Nexus file + m_instrument_name = nxload.getInstrumentName(); + // Read in the number of spectra in the Nexus file + m_numberOfSpectra = nxload.t_nsp1; + if(m_entrynumber!=0) + { + m_numberOfPeriods=1; + if(m_entrynumber>nxload.t_nper) + { + throw std::invalid_argument("Invalid Entry Number:Enter a valid number"); + } + } + else + { + // Read the number of periods in this file + m_numberOfPeriods = nxload.t_nper; + } + // Need to extract the user-defined output workspace name + Property *ws = getProperty("OutputWorkspace"); + std::string localWSName = ws->value(); + // If multiperiod, will need to hold the Instrument, Sample & SpectraDetectorMap for copying + boost::shared_ptr<Instrument> instrument; + boost::shared_ptr<Sample> sample; + + // Call private method to validate the optional parameters, if set + checkOptionalProperties(); + + // Read the number of time channels (i.e. bins) from the Nexus file + const int channelsPerSpectrum = nxload.t_ntc1; + // Read in the time bin boundaries + const int lengthIn = channelsPerSpectrum + 1; + float* timeChannels = new float[lengthIn]; + nxload.getTimeChannels(timeChannels, lengthIn); + // Put the read in array into a vector (inside a shared pointer) + boost::shared_ptr<MantidVec> timeChannelsVec + (new MantidVec(timeChannels, timeChannels + lengthIn)); + + // Calculate the size of a workspace, given its number of periods & spectra to read + int64_t total_specs; + if( m_interval || m_list) + { + total_specs = m_spec_list.size(); + if (m_interval) + { + total_specs += (m_spec_max-m_spec_min+1); + m_spec_max += 1; + } + } + else + { + total_specs = m_numberOfSpectra; + // for nexus return all spectra + m_spec_min = 0; // changed to 0 for NeXus, was 1 for Raw + m_spec_max = m_numberOfSpectra; // was +1? + } + + // Create the 2D workspace for the output + DataObjects::Workspace2D_sptr localWorkspace = boost::dynamic_pointer_cast<DataObjects::Workspace2D> + (WorkspaceFactory::Instance().create("Workspace2D",total_specs,lengthIn,lengthIn-1)); + localWorkspace->setTitle(title); + localWorkspace->setComment(notes); + localWorkspace->mutableRun().addLogData(new PropertyWithValue<std::string>("run_number", run_num)); + // Set the unit on the workspace to muon time, for now in the form of a Label Unit + boost::shared_ptr<Kernel::Units::Label> lblUnit = + boost::dynamic_pointer_cast<Kernel::Units::Label>(UnitFactory::Instance().create("Label")); + lblUnit->setLabel("Time","microsecond"); + localWorkspace->getAxis(0)->unit() = lblUnit; + // Set y axis unit + localWorkspace->setYUnit("Counts"); + + WorkspaceGroup_sptr wsGrpSptr=WorkspaceGroup_sptr(new WorkspaceGroup); + if(m_numberOfPeriods>1) + { + setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(wsGrpSptr)); + } + + API::Progress progress(this,0.,1.,m_numberOfPeriods * total_specs); + // Loop over the number of periods in the Nexus file, putting each period in a separate workspace + for (int64_t period = 0; period < m_numberOfPeriods; ++period) { + if(m_entrynumber!=0) + { + period=m_entrynumber-1; + if(period!=0) + { + loadRunDetails(localWorkspace); + runLoadInstrument(localWorkspace ); + runLoadMappingTable(localWorkspace ); + } + } + + if (period == 0) + { + // Only run the Child Algorithms once + loadRunDetails(localWorkspace); + runLoadInstrument(localWorkspace ); + runLoadMappingTable(localWorkspace ); + runLoadLog(localWorkspace ); + localWorkspace->populateInstrumentParameters(); + } + else // We are working on a higher period of a multiperiod raw file + { + localWorkspace = boost::dynamic_pointer_cast<DataObjects::Workspace2D> + (WorkspaceFactory::Instance().create(localWorkspace)); + localWorkspace->setTitle(title); + localWorkspace->setComment(notes); + //localWorkspace->newInstrumentParameters(); ??? + + } + + + std::string outws(""); + if(m_numberOfPeriods>1) + { + std::string outputWorkspace = "OutputWorkspace"; + std::stringstream suffix; + suffix << (period+1); + outws =outputWorkspace+"_"+suffix.str(); + std::string WSName = localWSName + "_" + suffix.str(); + declareProperty(new WorkspaceProperty<Workspace>(outws,WSName,Direction::Output)); + if (wsGrpSptr) + { + wsGrpSptr->addWorkspace( localWorkspace ); + } + } + + size_t counter = 0; + for (int64_t i = m_spec_min; i < m_spec_max; ++i) + { + // Shift the histogram to read if we're not in the first period + specid_t histToRead = static_cast<specid_t>(i + period*total_specs); + loadData(timeChannelsVec,counter,histToRead,nxload,lengthIn-1,localWorkspace ); // added -1 for NeXus + counter++; + progress.report(); + } + // Read in the spectra in the optional list parameter, if set + if (m_list) + { + for(size_t i=0; i < m_spec_list.size(); ++i) + { + loadData(timeChannelsVec,counter,m_spec_list[i],nxload,lengthIn-1, localWorkspace ); + counter++; + progress.report(); + } + } + // Just a sanity check + assert(counter == size_t(total_specs) ); + + bool autogroup = getProperty("AutoGroup"); + + if (autogroup) + { + + //Get the groupings + int64_t max_group = 0; + // use a map for mapping group number and output workspace index in case + // there are group numbers > number of groups + std::map<int64_t,int64_t> groups; + m_groupings.resize(nxload.numDetectors); + bool thereAreZeroes = false; + for (int64_t i =0; i < static_cast<int64_t>(nxload.numDetectors); ++i) + { + int64_t ig = static_cast<int64_t>(nxload.detectorGroupings[i]); + if (ig == 0) + { + thereAreZeroes = true; + continue; + } + m_groupings[i] = static_cast<specid_t>(ig); + if (groups.find(ig) == groups.end()) + groups[ig] = static_cast<int64_t>(groups.size()); + if (ig > max_group) max_group = ig; + } + + if (thereAreZeroes) + for (int64_t i =0; i < static_cast<int64_t>(nxload.numDetectors); ++i) + { + int64_t ig = static_cast<int64_t>(nxload.detectorGroupings[i]); + if (ig == 0) + { + ig = ++max_group; + m_groupings[i] = static_cast<specid_t>(ig); + groups[ig] = groups.size(); + } + } + + int numHists = static_cast<int>(localWorkspace->getNumberHistograms()); + size_t ngroups = groups.size(); // number of groups + + // to output groups in ascending order + { + int64_t i=0; + for(std::map<int64_t,int64_t>::iterator it=groups.begin();it!=groups.end();++it,++i) + { + it->second = i; + g_log.information()<<"group "<<it->first<<": "; + bool first = true; + int64_t first_i = -1 * std::numeric_limits<int64_t>::max(); + int64_t last_i = -1 * std::numeric_limits<int64_t>::max(); + for(int64_t i=0;i<static_cast<int64_t>(numHists);i++) + if (m_groupings[i] == it->first) + { + if (first) + { + first = false; + g_log.information()<<i; + first_i = i; + } + else + { + if (first_i >= 0) + { + if (i > last_i + 1) + { + g_log.information()<<'-'<<i; + first_i = -1; + } + } + else + { + g_log.information()<<','<<i; + first_i = i; + } + } + last_i = i; + } + else + { + if (!first && first_i >= 0) + { + if (last_i > first_i) + g_log.information()<<'-'<<last_i; + first_i = -1; + } + } + if (first_i >= 0 && last_i > first_i) + g_log.information()<<'-'<<last_i; + g_log.information()<<'\n'; + } + } + + //Create a workspace with only two spectra for forward and back + DataObjects::Workspace2D_sptr groupedWS = boost::dynamic_pointer_cast<DataObjects::Workspace2D> + (API::WorkspaceFactory::Instance().create(localWorkspace, ngroups, localWorkspace->dataX(0).size(), localWorkspace->blocksize())); + + boost::shared_array<specid_t> spec(new specid_t[numHists]); + boost::shared_array<detid_t> dets(new detid_t[numHists]); + + //Compile the groups + for (int i = 0; i < numHists; ++i) + { + specid_t k = static_cast<specid_t>(groups[ m_groupings[numHists*period + i] ]); + + for (detid_t j = 0; j < static_cast<detid_t>(localWorkspace->blocksize()); ++j) + { + groupedWS->dataY(k)[j] = groupedWS->dataY(k)[j] + localWorkspace->dataY(i)[j]; + + //Add the errors in quadrature + groupedWS->dataE(k)[j] + = sqrt(pow(groupedWS->dataE(k)[j], 2) + pow(localWorkspace->dataE(i)[j], 2)); + } + + //Copy all the X data + groupedWS->dataX(k) = localWorkspace->dataX(i); + spec[i] = k + 1; + dets[i] = i + 1; + } + + m_groupings.clear(); + + // All two spectra + for(detid_t k=0; k<static_cast<detid_t>(ngroups); k++) + { + groupedWS->getAxis(1)->spectraNo(k)= k + 1; + } + + groupedWS->replaceSpectraMap(new API::SpectraDetectorMap(spec.get(),dets.get(),numHists)); + + // Assign the result to the output workspace property + if(m_numberOfPeriods>1) + setProperty(outws, boost::dynamic_pointer_cast<Workspace>(groupedWS)); + else + { + setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(groupedWS)); + + } + + } + else + { + // Assign the result to the output workspace property + if(m_numberOfPeriods>1) + setProperty(outws,boost::dynamic_pointer_cast<Workspace>(localWorkspace)); + else + { + setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(localWorkspace)); + } + + } + + } // loop over periods + + // Clean up + delete[] timeChannels; + } + + /** Load in a single spectrum taken from a NeXus file + * @param tcbs :: The vector containing the time bin boundaries + * @param hist :: The workspace index + * @param i :: The spectrum number + * @param nxload :: A reference to the MuonNeXusReader object + * @param lengthIn :: The number of elements in a spectrum + * @param localWorkspace :: A pointer to the workspace in which the data will be stored + */ + void LoadMuonNexus1::loadData(const MantidVecPtr::ptr_type& tcbs,size_t hist, specid_t& i, + MuonNexusReader& nxload, const int64_t lengthIn, DataObjects::Workspace2D_sptr localWorkspace) + { + // Read in a spectrum + // Put it into a vector, discarding the 1st entry, which is rubbish + // But note that the last (overflow) bin is kept + // For Nexus, not sure if above is the case, hence give all data for now + MantidVec& Y = localWorkspace->dataY(hist); + Y.assign(nxload.counts + i * lengthIn, nxload.counts + i * lengthIn + lengthIn); + // Create and fill another vector for the errors, containing sqrt(count) + MantidVec& E = localWorkspace->dataE(hist); + typedef double (*uf)(double); + uf dblSqrt = std::sqrt; + std::transform(Y.begin(), Y.end(), E.begin(), dblSqrt); + // Populate the workspace. Loop starts from 1, hence i-1 + localWorkspace->setX(hist, tcbs); + localWorkspace->getAxis(1)->spectraNo(hist)= static_cast<int>(hist) + 1; + } + + + /** Log the run details from the file + * @param localWorkspace :: The workspace details to use + */ + void LoadMuonNexus1::loadRunDetails(DataObjects::Workspace2D_sptr localWorkspace) + { + API::Run & runDetails = localWorkspace->mutableRun(); + + runDetails.addProperty("run_title", localWorkspace->getTitle(), true); + + int numSpectra = static_cast<int>(localWorkspace->getNumberHistograms()); + runDetails.addProperty("nspectra", numSpectra); + + NXRoot root(m_filename); + try + { + std::string start_time = root.getString("run/start_time"); + runDetails.addProperty("run_start", start_time); + } + catch (std::runtime_error &) + { + g_log.warning("run/start_time is not available, run_start log not added."); + } + + + try + { + std::string stop_time = root.getString("run/stop_time"); + runDetails.addProperty("run_end", stop_time); + } + catch (std::runtime_error &) + { + g_log.warning("run/stop_time is not available, run_end log not added."); + } + + try + { + std::string dur = root.getString("run/duration"); + runDetails.addProperty("dur", dur); + runDetails.addProperty("durunits", 1); // 1 means second here + runDetails.addProperty("dur_secs", dur); + } + catch (std::runtime_error &) + { + g_log.warning("run/duration is not available, dur log not added."); + } + + // Get number of good frames + NXEntry runInstrumentBeam = root.openEntry("run/instrument/beam"); + NXInfo infoGoodTotalFrames = runInstrumentBeam.getDataSetInfo("frames_good"); + if (infoGoodTotalFrames.stat != NX_ERROR) + { + int dum = root.getInt("run/instrument/beam/frames_good"); + runDetails.addProperty("goodfrm", dum); + } + + } + + /// Run LoadInstrumentFromNexus as a Child Algorithm (only if loading from instrument definition file fails) + void LoadMuonNexus1::runLoadInstrumentFromNexus(DataObjects::Workspace2D_sptr localWorkspace) + { + g_log.information() << "Instrument definition file not found. Attempt to load information about \n" + << "the instrument from nexus data file.\n"; + + IAlgorithm_sptr loadInst = createChildAlgorithm("LoadInstrumentFromNexus"); + + // Now execute the Child Algorithm. Catch and log any error, but don't stop. + bool executionSuccessful(true); + try + { + loadInst->setPropertyValue("Filename", m_filename); + loadInst->setProperty<MatrixWorkspace_sptr> ("Workspace", localWorkspace); + loadInst->execute(); + } + catch( std::invalid_argument&) + { + g_log.information("Invalid argument to LoadInstrument Child Algorithm"); + executionSuccessful = false; + } + catch (std::runtime_error&) + { + g_log.information("Unable to successfully run LoadInstrument Child Algorithm"); + executionSuccessful = false; + } + + if ( !executionSuccessful ) g_log.error("No instrument definition loaded"); + } + + /// Run the LoadMappingTable Child Algorithm to fill the SpectraToDetectorMap + void LoadMuonNexus1::runLoadMappingTable(DataObjects::Workspace2D_sptr localWorkspace) + { + NXRoot root(m_filename); + NXInt number = root.openNXInt("run/instrument/detector/number"); + number.load(); + detid_t ndet = static_cast<detid_t>(number[0]/m_numberOfPeriods); + boost::shared_array<detid_t> det(new detid_t[ndet]); + for(detid_t i=0;i<ndet;i++) + { + det[i] = i + 1; + } + localWorkspace->replaceSpectraMap(new API::SpectraDetectorMap(det.get(),det.get(),ndet)); + } + + /// Run the LoadLog Child Algorithm + void LoadMuonNexus1::runLoadLog(DataObjects::Workspace2D_sptr localWorkspace) + { + IAlgorithm_sptr loadLog = createChildAlgorithm("LoadMuonLog"); + // Pass through the same input filename + loadLog->setPropertyValue("Filename",m_filename); + // Set the workspace property to be the same one filled above + loadLog->setProperty<MatrixWorkspace_sptr>("Workspace",localWorkspace); + + // Now execute the Child Algorithm. Catch and log any error, but don't stop. + try + { + loadLog->execute(); + } + catch (std::runtime_error&) + { + g_log.error("Unable to successfully run LoadLog Child Algorithm"); + } + catch (std::logic_error&) + { + g_log.error("Unable to successfully run LoadLog Child Algorithm"); + } + + if ( ! loadLog->isExecuted() ) g_log.error("Unable to successfully run LoadLog Child Algorithm"); + + + NXRoot root(m_filename); + + try + { + NXChar orientation = root.openNXChar("run/instrument/detector/orientation"); + // some files have no data there + orientation.load(); + + if (orientation[0] == 't') + { + Kernel::TimeSeriesProperty<double>* p = new Kernel::TimeSeriesProperty<double>("fromNexus"); + std::string start_time = root.getString("run/start_time"); + p->addValue(start_time,-90.0); + localWorkspace->mutableRun().addLogData(p); + setProperty("MainFieldDirection", "Transverse"); + } + else + { + setProperty("MainFieldDirection", "Longitudinal"); + } + } + catch(...) + { + setProperty("MainFieldDirection", "Longitudinal"); + } + + } + + /**checks the file by opening it and reading few lines + * @param filePath :: name of the file inluding its path + * @return an integer value how much this algorithm can load the file + */ + int LoadMuonNexus1::fileCheck(const std::string& filePath) + { + try + { + NXRoot root(filePath); + NXEntry entry = root.openFirstEntry(); + if ( ! entry.containsDataSet( "analysis" ) ) return 0; + std::string versionField = "IDF_version"; + if ( ! entry.containsDataSet( versionField ) ) + { + versionField = "idf_version"; + if ( ! entry.containsDataSet( versionField ) ) return 0; + } + if ( entry.getInt( versionField ) != 1 ) return 0; + std::string definition = entry.getString( "analysis" ); + if ( definition == "muonTD" || definition == "pulsedTD" ) + { + // If all this succeeded then we'll assume this is an ISIS Muon NeXus file version 1 + return 81; + } + } + catch(...) + { + } + return 0; + } + + } // namespace DataHandling +} // namespace Mantid diff --git a/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus2.cpp b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus2.cpp index 0ad28d34e65..3a7eeba29f7 100644 --- a/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus2.cpp +++ b/Code/Mantid/Framework/DataHandling/src/LoadMuonNexus2.cpp @@ -2,21 +2,21 @@ // Includes //---------------------------------------------------------------------- #include "MantidDataHandling/LoadMuonNexus2.h" +#include "MantidDataHandling/LoadMuonNexus1.h" #include "MantidDataObjects/Workspace2D.h" -#include "MantidKernel/UnitFactory.h" -#include "MantidKernel/ConfigService.h" -#include "MantidKernel/ArrayProperty.h" #include "MantidAPI/FileProperty.h" -#include "MantidKernel/TimeSeriesProperty.h" -#include "MantidGeometry/Instrument/Detector.h" #include "MantidAPI/Progress.h" #include "MantidAPI/SpectraDetectorMap.h" +#include "MantidAPI/LoadAlgorithmFactory.h" +#include "MantidGeometry/Instrument/Detector.h" +#include "MantidKernel/TimeSeriesProperty.h" +#include "MantidKernel/UnitFactory.h" +#include "MantidKernel/ConfigService.h" +#include "MantidKernel/ArrayProperty.h" #include "MantidNexus/NexusClasses.h" #include "MantidNexusCPP/NeXusFile.hpp" #include "MantidNexusCPP/NeXusException.hpp" - -#include "MantidAPI/LoadAlgorithmFactory.h" #include <Poco/Path.h> #include <boost/lexical_cast.hpp> #include <boost/shared_ptr.hpp> @@ -40,8 +40,8 @@ namespace Mantid /// Sets documentation strings for this algorithm void LoadMuonNexus2::initDocs() { - this->setWikiSummary("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 1 and use the results to populate the named workspace. LoadMuonNexus may be invoked by [[LoadNexus]] if it is given a NeXus file of this type. "); - this->setOptionalMessage("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 1 and use the results to populate the named workspace. LoadMuonNexus may be invoked by LoadNexus if it is given a NeXus file of this type."); + this->setWikiSummary("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 2 and use the results to populate the named workspace. LoadMuonNexus may be invoked by [[LoadNexus]] if it is given a NeXus file of this type. "); + this->setOptionalMessage("The LoadMuonNexus algorithm will read the given NeXus Muon data file Version 2 and use the results to populate the named workspace. LoadMuonNexus may be invoked by LoadNexus if it is given a NeXus file of this type."); } @@ -49,13 +49,49 @@ namespace Mantid LoadMuonNexus2::LoadMuonNexus2() : LoadMuonNexus() {} - /** Executes the algorithm. Reading in the file and creating and populating - * the output workspace + /** Executes the right version of the muon nexus loader: versions 1 or 2. * * @throw Exception::FileError If the Nexus file cannot be found/opened * @throw std::invalid_argument If the optional properties are set to invalid values */ void LoadMuonNexus2::exec() + { + std::string filePath = getPropertyValue("Filename"); + LoadMuonNexus1 load1; load1.initialize(); + + int confidence1 = load1.fileCheck( filePath ); + int confidence2 = this->fileCheck( filePath ); + + // if none can load the file throw + if ( confidence1 < 80 && confidence2 < 80) + { + throw Kernel::Exception::FileError("Cannot open the file ", filePath); + } + + if ( confidence2 > confidence1 ) + { + // this loader + doExec(); + } + else + { + // version 1 loader + IAlgorithm_sptr childAlg = createChildAlgorithm("LoadMuonNexus",0,1,true,1); + auto version1Loader = boost::dynamic_pointer_cast<API::Algorithm>( childAlg ); + version1Loader->copyPropertiesFrom( *this ); + version1Loader->executeAsChildAlg(); + this->copyPropertiesFrom( *version1Loader ); + API::Workspace_sptr outWS = version1Loader->getProperty("OutputWorkspace"); + setProperty("OutputWorkspace", outWS); + } + } + + /** Read in a muon nexus file of the version 2. + * + * @throw Exception::FileError If the Nexus file cannot be found/opened + * @throw std::invalid_argument If the optional properties are set to invalid values + */ + void LoadMuonNexus2::doExec() { // Create the root Nexus class NXRoot root(getPropertyValue("Filename")); @@ -67,45 +103,27 @@ namespace Mantid } // Open the data entry - std::string entryName = root.groups()[iEntry].nxname; - NXEntry entry = root.openEntry(entryName); + m_entry_name = root.groups()[iEntry].nxname; + NXEntry entry = root.openEntry(m_entry_name); + + // Read in the instrument name from the Nexus file + m_instrument_name = entry.getString("instrument/name"); - NXInfo info = entry.getDataSetInfo("definition"); - if (info.stat == NX_ERROR) + // Read the number of periods in this file + if ( entry.containsGroup("run") ) { - info = entry.getDataSetInfo("analysis"); - std::string compareString = "muon"; - if (info.stat == NX_OK && entry.getString("analysis").compare(0,compareString.length(),compareString) == 0) + try { - LoadMuonNexus::exec(); - return; + m_numberOfPeriods = entry.getInt("run/number_periods"); } - else + catch (::NeXus::Exception&) { - g_log.debug()<<"analysis=|" << entry.getString("analysis") << "|" << std::endl; - throw std::runtime_error("Unknown Muon NeXus file format"); + //assume 1 + m_numberOfPeriods = 1; } } else { - std::string definition = entry.getString("definition"); - if (info.stat == NX_ERROR || definition != "pulsedTD") - { - throw std::runtime_error("Unknown Muon Nexus file format"); - } - } - - // Read in the instrument name from the Nexus file - m_instrument_name = entry.getString("instrument/name"); - - // Read the number of periods in this file - try - { - m_numberOfPeriods = entry.getInt("run/number_periods"); - } - catch (::NeXus::Exception&) - { - //assume 1 m_numberOfPeriods = 1; } @@ -179,56 +197,44 @@ namespace Mantid //g_log.error()<<" number of perioids= "<<m_numberOfPeriods<<std::endl; WorkspaceGroup_sptr wsGrpSptr=WorkspaceGroup_sptr(new WorkspaceGroup); - try + if ( entry.containsDataSet( "title" ) ) { wsGrpSptr->setTitle(entry.getString("title")); } - catch (::NeXus::Exception&) - {} - try + + if ( entry.containsDataSet( "notes" ) ) { wsGrpSptr->setComment(entry.getString("notes")); } - catch (::NeXus::Exception&) - {} if(m_numberOfPeriods>1) { setProperty("OutputWorkspace",boost::dynamic_pointer_cast<Workspace>(wsGrpSptr)); } - Mantid::NeXus::NXInt period_index = dataGroup.openNXInt("period_index"); - period_index.load(); + // period_index is currently unused + //Mantid::NeXus::NXInt period_index = dataGroup.openNXInt("period_index"); + //period_index.load(); Mantid::NeXus::NXInt counts = dataGroup.openIntData(); counts.load(); + NXInstrument instr = entry.openNXInstrument("instrument"); - try + if ( instr.containsGroup( "detector_fb" ) ) { - NXEntry entryTimeZero = root.openEntry("run/instrument/detector_fb"); - NXInfo infoTimeZero = entryTimeZero.getDataSetInfo("time_zero"); - if (infoTimeZero.stat != NX_ERROR) + NXDetector detector = instr.openNXDetector("detector_fb"); + if (detector.containsDataSet("time_zero")) { - double dum = root.getFloat("run/instrument/detector_fb/time_zero"); + double dum = detector.getFloat("time_zero"); setProperty("TimeZero", dum); } - } - catch (::NeXus::Exception&) - {} - - try - { - NXEntry entryFGB = root.openEntry("run/instrument/detector_fb"); - NXInfo infoFGB = entryFGB.getDataSetInfo("first_good_time"); - if (infoFGB.stat != NX_ERROR) + if (detector.containsDataSet("first_good_time")) { - double dum = root.getFloat("run/instrument/detector_fb/first_good_time"); + double dum = detector.getFloat("first_good_time"); setProperty("FirstGoodData", dum); } } - catch (::NeXus::Exception&) - {} API::Progress progress(this,0.,1.,m_numberOfPeriods * total_specs); // Loop over the number of periods in the Nexus file, putting each period in a separate workspace @@ -321,10 +327,27 @@ namespace Mantid MantidVec& X = localWorkspace->dataX(wsIndex); MantidVec& Y = localWorkspace->dataY(wsIndex); MantidVec& E = localWorkspace->dataE(wsIndex); - int nBins = counts.dim2(); - assert( nBins+1 == static_cast<int>(timeBins.size()) ); X.assign(timeBins.begin(),timeBins.end()); - int *data = &counts(period,spec,0); + + int nBins = 0; + int *data = NULL; + + if ( counts.rank() == 3 ) + { + nBins = counts.dim2(); + data = &counts(period,spec,0); + } + else if ( counts.rank() == 2 ) + { + nBins = counts.dim1(); + data = &counts(spec,0); + } + else + { + throw std::runtime_error("Data have unsupported dimansionality"); + } + assert( nBins+1 == static_cast<int>(timeBins.size()) ); + Y.assign(data,data+nBins); typedef double (*uf)(double); uf dblSqrt = std::sqrt; @@ -357,7 +380,11 @@ namespace Mantid } ws->setTitle(entry.getString("title")); - ws->setComment(entry.getString("notes")); + + if ( entry.containsDataSet("notes") ) + { + ws->setComment(entry.getString("notes")); + } std::string run_num = boost::lexical_cast<std::string>(entry.getInt("run_number")); //The sample is left to delete the property @@ -366,86 +393,6 @@ namespace Mantid ws->populateInstrumentParameters(); } - /**This method does a quick file type check by looking at the first 100 bytes of the file - * @param filePath- path of the file including name. - * @param nread :: no.of bytes read - * @param header :: The first 100 bytes of the file as a union - * @return true if the given file is of type which can be loaded by this algorithm - */ - bool LoadMuonNexus2::quickFileCheck(const std::string& filePath,size_t nread,const file_header& header) - { - std::string extn=extension(filePath); - bool bnexs(false); - (!extn.compare("nxs")||!extn.compare("nx5"))?bnexs=true:bnexs=false; - /* - * HDF files have magic cookie in the first 4 bytes - */ - if ( ((nread >= sizeof(unsigned)) && (ntohl(header.four_bytes) == g_hdf_cookie)) || bnexs ) - { - //hdf - return true; - } - else if ( (nread >= sizeof(g_hdf5_signature)) && (!memcmp(header.full_hdr, g_hdf5_signature, sizeof(g_hdf5_signature))) ) - { - //hdf5 - return true; - } - return false; - - } - /**checks the file by opening it and reading few lines - * @param filePath :: name of the file inluding its path - * @return an integer value how much this algorithm can load the file - */ - int LoadMuonNexus2::fileCheck(const std::string& filePath) - { - int confidence(0); - std::string analysisType; - try - { - ::NeXus::File file = ::NeXus::File(filePath); - // Will throw if this doesn't exist - file.openPath("/run/analysis"); - analysisType = file.getStrData(); - } - catch(::NeXus::Exception&) - { - try - { - ::NeXus::File file = ::NeXus::File(filePath); - file.openPath("/run/definition"); - analysisType = file.getStrData(); - } - catch(::NeXus::Exception&) - { - //one last try - this is the area for PSI written files - try - { - ::NeXus::File file = ::NeXus::File(filePath); - file.openPath("/raw_data_1/definition"); - analysisType = file.getStrData(); - } - catch(::NeXus::Exception&) - { - //no give up this doesn't look like a muon v2 file - analysisType = ""; - } - } - } - std::string compareString = "muon"; - if( analysisType == "pulsedTD" ) - { - confidence = 85; - } - else if( analysisType.compare(0,compareString.length(),compareString) == 0 ) - { - confidence = 50; - } - else confidence = 0; - - return confidence; - } - /** Log the run details from the file * @param localWorkspace :: The workspace details to use */ @@ -460,28 +407,29 @@ namespace Mantid m_filename = getPropertyValue("Filename"); NXRoot root(m_filename); + NXEntry entry = root.openEntry(m_entry_name); - std::string start_time = root.getString("run/start_time"); + std::string start_time = entry.getString("start_time"); runDetails.addProperty("run_start", start_time); - std::string stop_time = root.getString("run/end_time"); + std::string stop_time = entry.getString("end_time"); runDetails.addProperty("run_end", stop_time); - - NXEntry runRun = root.openEntry("run/run"); - - NXInfo infoGoodTotalFrames = runRun.getDataSetInfo("good_total_frames"); - if (infoGoodTotalFrames.stat != NX_ERROR) + if ( entry.containsGroup( "run" ) ) { - int dum = root.getInt("run/run/good_total_frames"); - runDetails.addProperty("goodfrm", dum); - } + NXClass runRun = entry.openNXGroup("run"); - NXInfo infoNumberPeriods = runRun.getDataSetInfo("number_periods"); - if (infoNumberPeriods.stat != NX_ERROR) - { - int dum = root.getInt("run/run/number_periods"); - runDetails.addProperty("nperiods", dum); + if ( runRun.containsDataSet("good_total_frames") ) + { + int dum = runRun.getInt("good_total_frames"); + runDetails.addProperty("goodfrm", dum); + } + + if ( runRun.containsDataSet("number_periods") ) + { + int dum = runRun.getInt("number_periods"); + runDetails.addProperty("nperiods", dum); + } } { // Duration taken to be stop_time minus stat_time @@ -490,9 +438,37 @@ namespace Mantid double duration_in_secs = DateAndTime::secondsFromDuration( end - start); runDetails.addProperty("dur_secs",duration_in_secs); } + } - - + /**checks the file by opening it and reading few lines + * @param filePath :: name of the file inluding its path + * @return an integer value how much this algorithm can load the file + */ + int LoadMuonNexus2::fileCheck(const std::string& filePath) + { + try + { + NXRoot root(filePath); + NXEntry entry = root.openFirstEntry(); + if ( ! entry.containsDataSet( "definition" ) ) return 0; + std::string versionField = "IDF_version"; + if ( ! entry.containsDataSet( versionField ) ) + { + versionField = "idf_version"; + if ( ! entry.containsDataSet( versionField ) ) return 0; + } + if ( entry.getInt( versionField ) != 2 ) return 0; + std::string definition = entry.getString( "definition" ); + if ( definition == "muonTD" || definition == "pulsedTD" ) + { + // If all this succeeded then we'll assume this is an ISIS Muon NeXus file version 2 + return 81; + } + } + catch( ... ) + { + } + return 0; } } // namespace DataHandling diff --git a/Code/Mantid/Framework/DataHandling/test/GroupDetectors2Test.h b/Code/Mantid/Framework/DataHandling/test/GroupDetectors2Test.h index c6f55455d95..70bf0c22b77 100644 --- a/Code/Mantid/Framework/DataHandling/test/GroupDetectors2Test.h +++ b/Code/Mantid/Framework/DataHandling/test/GroupDetectors2Test.h @@ -12,7 +12,7 @@ #include "MantidKernel/ArrayProperty.h" #include "MantidKernel/Exception.h" #include "MantidKernel/UnitFactory.h" -#include "MantidDataHandling/LoadMuonNexus2.h" +#include "MantidDataHandling/LoadMuonNexus1.h" #include "MantidTestHelpers/WorkspaceCreationHelper.h" #include <cxxtest/TestSuite.h> #include <fstream> @@ -351,7 +351,7 @@ public: void testReadingFromXML() { - Mantid::DataHandling::LoadMuonNexus2 nxLoad; + Mantid::DataHandling::LoadMuonNexus1 nxLoad; nxLoad.initialize(); // Now set required filename and output workspace name @@ -391,7 +391,7 @@ public: void testReadingFromXMLCheckDublicateIndex() { - Mantid::DataHandling::LoadMuonNexus2 nxLoad; + Mantid::DataHandling::LoadMuonNexus1 nxLoad; nxLoad.initialize(); // Now set required filename and output workspace name @@ -431,7 +431,7 @@ public: void testReadingFromXMLCheckDublicateIndex2() { - Mantid::DataHandling::LoadMuonNexus2 nxLoad; + Mantid::DataHandling::LoadMuonNexus1 nxLoad; nxLoad.initialize(); // Now set required filename and output workspace name diff --git a/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus1Test.h b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus1Test.h new file mode 100644 index 00000000000..0e533d3d8cd --- /dev/null +++ b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus1Test.h @@ -0,0 +1,415 @@ +#ifndef LOADMUONNEXUS1TEST_H_ +#define LOADMUONNEXUS1TEST_H_ + + + +// These includes seem to make the difference between initialization of the +// workspace names (workspace2D/1D etc), instrument classes and not for this test case. +#include "MantidDataObjects/WorkspaceSingleValue.h" +#include "MantidDataHandling/LoadInstrument.h" +// + +#include <fstream> +#include <cxxtest/TestSuite.h> + +#include "MantidDataHandling/LoadMuonNexus1.h" +#include "MantidAPI/WorkspaceFactory.h" +#include "MantidDataObjects/ManagedWorkspace2D.h" +#include "MantidAPI/AnalysisDataService.h" +#include "MantidAPI/FrameworkManager.h" +#include "MantidKernel/ConfigService.h" +#include "MantidKernel/TimeSeriesProperty.h" +#include "MantidAPI/SpectraDetectorMap.h" +#include <Poco/Path.h> + +using namespace Mantid::API; +using namespace Mantid::Kernel; +using namespace Mantid::DataHandling; +using namespace Mantid::DataObjects; + +class LoadMuonNexus1Test : public CxxTest::TestSuite +{ +public: + + + void testInit() + { + TS_ASSERT_THROWS_NOTHING(nxLoad.initialize()); + TS_ASSERT( nxLoad.isInitialized() ); + } + + + void testExec() + { + if ( !nxLoad.isInitialized() ) nxLoad.initialize(); + // Should fail because mandatory parameter has not been set + TS_ASSERT_THROWS(nxLoad.execute(),std::runtime_error); + + // Now set required filename and output workspace name + inputFile = "emu00006473.nxs"; + nxLoad.setPropertyValue("FileName", inputFile); + + outputSpace="outer"; + nxLoad.setPropertyValue("OutputWorkspace", outputSpace); + + // + // Test execute to read file and populate workspace + // + TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); + TS_ASSERT( nxLoad.isExecuted() ); + + // Test additional output parameters + std::string field = nxLoad.getProperty("MainFieldDirection"); + TS_ASSERT( field == "Longitudinal" ); + + // + // Test workspace data (copied from LoadRawTest.h) + // + MatrixWorkspace_sptr output; + output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace); + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006473.nxs"; + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); + // Check one particular value + TS_ASSERT_EQUALS( output2D->dataY(11)[686], 81); + // Check that the error on that value is correct + TS_ASSERT_EQUALS( output2D->dataE(11)[686], 9); + // Check that the time is as expected from bin boundary update + TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); + + // Check the unit has been set correctly + TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); + TS_ASSERT( ! output-> isDistribution() ); + + /* - other tests from LoadRawTest - These test data not in current Nexus files + //---------------------------------------------------------------------- + // Tests taken from LoadInstrumentTest to check Child Algorithm is running properly + //---------------------------------------------------------------------- + boost::shared_ptr<Instrument> i = output->getInstrument(); + Mantid::Geometry::Component* source = i->getSource(); + + TS_ASSERT_EQUALS( source->getName(), "undulator"); + TS_ASSERT_DELTA( source->getPos().Y(), 0.0,0.01); + + Mantid::Geometry::Component* samplepos = i->getSample(); + TS_ASSERT_EQUALS( samplepos->getName(), "nickel-holder"); + TS_ASSERT_DELTA( samplepos->getPos().Y(), 10.0,0.01); + + Mantid::Geometry::Detector *ptrDet103 = dynamic_cast<Mantid::Geometry::Detector*>(i->getDetector(103)); + TS_ASSERT_EQUALS( ptrDet103->getID(), 103); + TS_ASSERT_EQUALS( ptrDet103->getName(), "pixel"); + TS_ASSERT_DELTA( ptrDet103->getPos().X(), 0.4013,0.01); + TS_ASSERT_DELTA( ptrDet103->getPos().Z(), 2.4470,0.01); + */ + //---------------------------------------------------------------------- + // Test code copied from LoadLogTest to check Child Algorithm is running properly + //---------------------------------------------------------------------- + //boost::shared_ptr<Sample> sample = output->getSample(); + Property *l_property = output->run().getLogData( std::string("beamlog_current") ); + TimeSeriesProperty<double> *l_timeSeriesDouble = dynamic_cast<TimeSeriesProperty<double>*>(l_property); + std::string timeSeriesString = l_timeSeriesDouble->value(); + TS_ASSERT_EQUALS( timeSeriesString.substr(0,27), "2006-Nov-21 07:03:08 182.8" ); + //check that sample name has been set correctly + TS_ASSERT_EQUALS(output->sample().getName(), "Cr2.7Co0.3Si"); + + /* + //---------------------------------------------------------------------- + // Tests to check that Loading SpectraDetectorMap is done correctly + //---------------------------------------------------------------------- + map= output->getSpectraMap(); + + // Check the total number of elements in the map for HET + TS_ASSERT_EQUALS(map->nElements(),24964); + + // Test one to one mapping, for example spectra 6 has only 1 pixel + TS_ASSERT_EQUALS(map->ndet(6),1); + + // Test one to many mapping, for example 10 pixels contribute to spectra 2084 + TS_ASSERT_EQUALS(map->ndet(2084),10); + // Check the id number of all pixels contributing + std::vector<Mantid::Geometry::IDetector*> detectorgroup; + detectorgroup=map->getDetectors(2084); + std::vector<Mantid::Geometry::IDetector*>::iterator it; + int pixnum=101191; + for (it=detectorgroup.begin();it!=detectorgroup.end();it++) + TS_ASSERT_EQUALS((*it)->getID(),pixnum++); + + // Test with spectra that does not exist + // Test that number of pixel=0 + TS_ASSERT_EQUALS(map->ndet(5),0); + // Test that trying to get the Detector throws. + boost::shared_ptr<Mantid::Geometry::IDetector> test; + TS_ASSERT_THROWS(test=map->getDetector(5),std::runtime_error); + */ + + } +// void testWithManagedWorkspace() +// { +// ConfigService::Instance().updateConfig("UseManagedWS.properties"); +// //LoadRaw loader4; +// //loader4.initialize(); +// //loader4.setPropertyValue("Filename", inputFile); +// //loader4.setPropertyValue("OutputWorkspace", "managedws"); +// // TS_ASSERT_THROWS_NOTHING( loader4.execute() ) +// //TS_ASSERT( loader4.isExecuted() ) +// +// // Get back workspace and check it really is a ManagedWorkspace2D +// MatrixWorkspace_sptr output; +// TS_ASSERT_THROWS_NOTHING( output = AnalysisDataService::Instance().retrieve("managedws") ); +// TS_ASSERT( dynamic_cast<ManagedWorkspace2D*>(output.get()) ) +// } + + + + void testTransvereDataset() + { + LoadMuonNexus1 nxL; + if ( !nxL.isInitialized() ) nxL.initialize(); + + // Now set required filename and output workspace name + std::string inputFile_musr00022725 = "MUSR00022725.nxs"; + nxL.setPropertyValue("FileName", inputFile_musr00022725); + + outputSpace="outermusr00022725"; + nxL.setPropertyValue("OutputWorkspace", outputSpace); + + TS_ASSERT_THROWS_NOTHING(nxL.execute()); + TS_ASSERT( nxL.isExecuted() ); + + // Test additional output parameters + std::string field = nxL.getProperty("MainFieldDirection"); + TS_ASSERT( field == "Transverse" ); + double timeZero = nxL.getProperty("TimeZero"); + TS_ASSERT_DELTA( timeZero, 0.55,0.001); + double firstgood = nxL.getProperty("FirstGoodData"); + TS_ASSERT_DELTA( firstgood, 0.656,0.001); + std::vector<double> deadTimes = nxL.getProperty("DeadTimes"); + TS_ASSERT_DELTA( deadTimes[0], 0.006,0.001); + TS_ASSERT_DELTA( deadTimes[deadTimes.size()-1], 0.011,0.001); + } + + void testExec2() + { + //test for multi period + // Now set required filename and output workspace name + inputFile2 = "emu00006475.nxs"; + nxLoad.setPropertyValue("FileName", inputFile2); + + outputSpace="outer2"; + nxLoad.setPropertyValue("OutputWorkspace", outputSpace); + nxLoad.setPropertyValue("EntryNumber", "1"); + int64_t entryNumber=nxLoad.getProperty("EntryNumber"); + + // + // Test execute to read file and populate workspace + // + TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); + TS_ASSERT( nxLoad.isExecuted() ); + // + // Test workspace data - should be 4 separate workspaces for this 4 period file + // + if(entryNumber==0) + { + WorkspaceGroup_sptr outGrp; + TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); + + } + //if entry number is given + if(entryNumber==1) + { + MatrixWorkspace_sptr output; + output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace); + + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + //Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); + // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); + // Check that the time is as expected from bin boundary update + TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); + + // Check the unit has been set correctly + TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); + TS_ASSERT( ! output-> isDistribution() ); + + //check that sample name has been set correctly + //boost::shared_ptr<Sample> sample,sample2; + //sample = output->getSample(); + //sample2 = output2->getSample(); + //TS_ASSERT_EQUALS(sample->getName(), sample2->getName()); + TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); + + } + MatrixWorkspace_sptr output,output2,output3,output4; + WorkspaceGroup_sptr outGrp; + //if no entry number load the group workspace + if(entryNumber==0) + { + TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); + + (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_1")); + (output2 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_2")); + (output3 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_3")); + (output4 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_4")); + + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); + // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); + // Check one particular value + TS_ASSERT_EQUALS( output2D2->dataY(8)[502], 121); + // Check that the error on that value is correct + TS_ASSERT_EQUALS( output2D2->dataE(8)[502], 11); + // Check that the time is as expected from bin boundary update + TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); + + // Check the unit has been set correctly + TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); + TS_ASSERT( ! output-> isDistribution() ); + + //check that sample name has been set correctly + // boost::shared_ptr<Sample> sample,sample2; + //sample = output->getSample(); + // sample2 = output2->getSample(); + TS_ASSERT_EQUALS(output->sample().getName(), output2->sample().getName()); + TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); + + } + } + void testExec2withZeroEntryNumber() + { + //test for multi period + // Now set required filename and output workspace name + inputFile2 = "emu00006475.nxs"; + nxLoad.setPropertyValue("FileName", inputFile2); + + outputSpace="outer2"; + nxLoad.setPropertyValue("OutputWorkspace", outputSpace); + nxLoad.setPropertyValue("EntryNumber", "0"); + int64_t entryNumber=nxLoad.getProperty("EntryNumber"); + + // + // Test execute to read file and populate workspace + // + TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); + TS_ASSERT( nxLoad.isExecuted() ); + // + // Test workspace data - should be 4 separate workspaces for this 4 period file + // + WorkspaceGroup_sptr outGrp; + TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); + + MatrixWorkspace_sptr output,output2,output3,output4; + //WorkspaceGroup_sptr outGrp; + //if no entry number load the group workspace + if(entryNumber==0) + { + //TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); + + (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_1")); + (output2 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_2")); + (output3 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_3")); + (output4 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_4")); + + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); + // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); + // Check one particular value + TS_ASSERT_EQUALS( output2D2->dataY(8)[502], 121); + // Check that the error on that value is correct + TS_ASSERT_EQUALS( output2D2->dataE(8)[502], 11); + // Check that the time is as expected from bin boundary update + TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); + + // Check the unit has been set correctly + TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); + TS_ASSERT( ! output-> isDistribution() ); + + //check that sample name has been set correctly + //boost::shared_ptr<Sample> sample,sample2; + // sample = output->getSample(); + // sample2 = output2->getSample(); + TS_ASSERT_EQUALS(output->sample().getName(), output2->sample().getName()); + TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); + + } + } + + void testarrayin() + { + if ( !nxload3.isInitialized() ) nxload3.initialize(); + + nxload3.setPropertyValue("Filename", inputFile); + nxload3.setPropertyValue("OutputWorkspace", "outWS"); + nxload3.setPropertyValue("SpectrumList", "29,30,31"); + nxload3.setPropertyValue("SpectrumMin", "5"); + nxload3.setPropertyValue("SpectrumMax", "10"); + + TS_ASSERT_THROWS_NOTHING(nxload3.execute()); + TS_ASSERT( nxload3.isExecuted() ); + + // Get back the saved workspace + MatrixWorkspace_sptr output; + (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>("outWS")); + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + + // Should be 6 for selected input + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 9); + + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(1)) == (output2D->dataX(5)) ); + + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(2).size(), output2D->dataY(7).size() ); + + // Check one particular value + TS_ASSERT_EQUALS( output2D->dataY(8)[479], 144); + // Check that the error on that value is correct + TS_ASSERT_EQUALS( output2D->dataE(8)[479], 12); + // Check that the error on that value is correct + TS_ASSERT_DELTA( output2D->dataX(8)[479], 7.410, 0.0001); + } + +private: + LoadMuonNexus1 nxLoad,nxload2,nxload3; + std::string outputSpace; + std::string entryName; + std::string inputFile; + std::string inputFile2; + boost::shared_ptr<SpectraDetectorMap> map; +}; + +////------------------------------------------------------------------------------ +//// Performance test +////------------------------------------------------------------------------------ +// +//class LoadMuonNexus1TestPerformance : public CxxTest::TestSuite +//{ +//public: +// void testDefaultLoad() +// { +// LoadMuonNexus1 loader; +// loader.initialize(); +// loader.setPropertyValue("Filename", "deltat_tdc_gpd_0900.nxs"); +// loader.setPropertyValue("OutputWorkspace", "ws"); +// TS_ASSERT( loader.execute() ); +// } +//}; +// +#endif /*LOADMUONNEXUS1TEST_H_*/ diff --git a/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus2Test.h b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus2Test.h index a3bb3d00845..17909c0d7de 100644 --- a/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus2Test.h +++ b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexus2Test.h @@ -358,7 +358,7 @@ public: check_spectra_and_detectors(output); - AnalysisDataService::Instance().remove(outputSpace); + AnalysisDataService::Instance().clear(); } void testExec2() @@ -443,9 +443,69 @@ public: check_spectra_and_detectors(output); - AnalysisDataService::Instance().remove(outputSpace); + AnalysisDataService::Instance().clear(); } + void test_gpd_file() + { + LoadMuonNexus2 nxLoad; + nxLoad.initialize(); + + // Now set required filename and output workspace name + std::string inputFile = "deltat_tdc_gpd_0900.nxs"; + nxLoad.setPropertyValue("FileName", inputFile); + + std::string outputSpace="outer"; + nxLoad.setPropertyValue("OutputWorkspace", outputSpace); + + // + // Test execute to read file and populate workspace + // + TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); + TS_ASSERT( nxLoad.isExecuted() ); + + // + // Test additional output parameters + // + std::string field = nxLoad.getProperty("MainFieldDirection"); + TS_ASSERT( field == "Transverse" ); + // TimeZero and FirstGoodData are not read yet so they are 0 + double timeZero = nxLoad.getProperty("TimeZero"); + TS_ASSERT_DELTA( timeZero, 0.0,0.001); + double firstgood = nxLoad.getProperty("FirstGoodData"); + TS_ASSERT_DELTA( firstgood, 0.0,0.001); + + // + // Test workspace data + // + MatrixWorkspace_sptr output; + output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace); + Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); + // Should be 192 for file inputFile = "argus0026287.nxs"; + TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 2); + TS_ASSERT_EQUALS( output2D->blocksize(), 8192); + // Check two X vectors are the same + TS_ASSERT( (output2D->dataX(0)) == (output2D->dataX(1)) ); + // Check two Y arrays have the same number of elements + TS_ASSERT_EQUALS( output2D->dataY(0).size(), output2D->dataY(1).size() ); + // Check one particular value + TS_ASSERT_EQUALS( output2D->dataY(0)[686], 516); + TS_ASSERT_EQUALS( output2D->dataY(0)[687], 413); + TS_ASSERT_EQUALS( output2D->dataY(1)[686], 381); + + // Check that the error on that value is correct + TS_ASSERT_DELTA( output2D->dataE(0)[686], 22.7156,0.001); + TS_ASSERT_DELTA( output2D->dataE(0)[687], 20.3224,0.001); + TS_ASSERT_DELTA( output2D->dataE(1)[686], 19.5192,0.001); + // Check that the time is as expected from bin boundary update + TS_ASSERT_DELTA( output2D->dataX(1)[687], 0.8050,0.001); + + // Check the unit has been set correctly + TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); + TS_ASSERT( ! output-> isDistribution() ); + + AnalysisDataService::Instance().remove(outputSpace); + } }; //------------------------------------------------------------------------------ diff --git a/Code/Mantid/Framework/DataHandling/test/LoadMuonNexusTest.h b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexusTest.h index 6c62b27823a..3bbae100287 100644 --- a/Code/Mantid/Framework/DataHandling/test/LoadMuonNexusTest.h +++ b/Code/Mantid/Framework/DataHandling/test/LoadMuonNexusTest.h @@ -1,397 +1,20 @@ #ifndef LOADMUONNEXUSTEST_H_ #define LOADMUONNEXUSTEST_H_ - - -// These includes seem to make the difference between initialization of the -// workspace names (workspace2D/1D etc), instrument classes and not for this test case. -#include "MantidDataObjects/WorkspaceSingleValue.h" -#include "MantidDataHandling/LoadInstrument.h" -// - -#include <fstream> #include <cxxtest/TestSuite.h> -#include "MantidDataHandling/LoadMuonNexus.h" -#include "MantidAPI/WorkspaceFactory.h" -#include "MantidDataObjects/ManagedWorkspace2D.h" -#include "MantidAPI/AnalysisDataService.h" -#include "MantidAPI/FrameworkManager.h" -#include "MantidKernel/ConfigService.h" -#include "MantidKernel/TimeSeriesProperty.h" -#include "MantidAPI/SpectraDetectorMap.h" -#include <Poco/Path.h> +#include "MantidDataHandling/LoadMuonNexus3.h" -using namespace Mantid::API; -using namespace Mantid::Kernel; using namespace Mantid::DataHandling; -using namespace Mantid::DataObjects; class LoadMuonNexusTest : public CxxTest::TestSuite { public: - - void testInit() - { - TS_ASSERT_THROWS_NOTHING(nxLoad.initialize()); - TS_ASSERT( nxLoad.isInitialized() ); - } - - void testExec() { - if ( !nxLoad.isInitialized() ) nxLoad.initialize(); - // Should fail because mandatory parameter has not been set - TS_ASSERT_THROWS(nxLoad.execute(),std::runtime_error); - - // Now set required filename and output workspace name - inputFile = "emu00006473.nxs"; - nxLoad.setPropertyValue("FileName", inputFile); - - outputSpace="outer"; - nxLoad.setPropertyValue("OutputWorkspace", outputSpace); - - // - // Test execute to read file and populate workspace - // - TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); - TS_ASSERT( nxLoad.isExecuted() ); - - // Test additional output parameters - std::string field = nxLoad.getProperty("MainFieldDirection"); - TS_ASSERT( field == "Longitudinal" ); - - // - // Test workspace data (copied from LoadRawTest.h) - // - MatrixWorkspace_sptr output; - output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace); - Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); - // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006473.nxs"; - TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); - // Check two X vectors are the same - TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); - // Check two Y arrays have the same number of elements - TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); - // Check one particular value - TS_ASSERT_EQUALS( output2D->dataY(11)[686], 81); - // Check that the error on that value is correct - TS_ASSERT_EQUALS( output2D->dataE(11)[686], 9); - // Check that the time is as expected from bin boundary update - TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); - - // Check the unit has been set correctly - TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); - TS_ASSERT( ! output-> isDistribution() ); - - /* - other tests from LoadRawTest - These test data not in current Nexus files - //---------------------------------------------------------------------- - // Tests taken from LoadInstrumentTest to check Child Algorithm is running properly - //---------------------------------------------------------------------- - boost::shared_ptr<Instrument> i = output->getInstrument(); - Mantid::Geometry::Component* source = i->getSource(); - - TS_ASSERT_EQUALS( source->getName(), "undulator"); - TS_ASSERT_DELTA( source->getPos().Y(), 0.0,0.01); - - Mantid::Geometry::Component* samplepos = i->getSample(); - TS_ASSERT_EQUALS( samplepos->getName(), "nickel-holder"); - TS_ASSERT_DELTA( samplepos->getPos().Y(), 10.0,0.01); - - Mantid::Geometry::Detector *ptrDet103 = dynamic_cast<Mantid::Geometry::Detector*>(i->getDetector(103)); - TS_ASSERT_EQUALS( ptrDet103->getID(), 103); - TS_ASSERT_EQUALS( ptrDet103->getName(), "pixel"); - TS_ASSERT_DELTA( ptrDet103->getPos().X(), 0.4013,0.01); - TS_ASSERT_DELTA( ptrDet103->getPos().Z(), 2.4470,0.01); - */ - //---------------------------------------------------------------------- - // Test code copied from LoadLogTest to check Child Algorithm is running properly - //---------------------------------------------------------------------- - //boost::shared_ptr<Sample> sample = output->getSample(); - Property *l_property = output->run().getLogData( std::string("beamlog_current") ); - TimeSeriesProperty<double> *l_timeSeriesDouble = dynamic_cast<TimeSeriesProperty<double>*>(l_property); - std::string timeSeriesString = l_timeSeriesDouble->value(); - TS_ASSERT_EQUALS( timeSeriesString.substr(0,27), "2006-Nov-21 07:03:08 182.8" ); - //check that sample name has been set correctly - TS_ASSERT_EQUALS(output->sample().getName(), "Cr2.7Co0.3Si"); - - /* - //---------------------------------------------------------------------- - // Tests to check that Loading SpectraDetectorMap is done correctly - //---------------------------------------------------------------------- - map= output->getSpectraMap(); - - // Check the total number of elements in the map for HET - TS_ASSERT_EQUALS(map->nElements(),24964); - - // Test one to one mapping, for example spectra 6 has only 1 pixel - TS_ASSERT_EQUALS(map->ndet(6),1); - - // Test one to many mapping, for example 10 pixels contribute to spectra 2084 - TS_ASSERT_EQUALS(map->ndet(2084),10); - // Check the id number of all pixels contributing - std::vector<Mantid::Geometry::IDetector*> detectorgroup; - detectorgroup=map->getDetectors(2084); - std::vector<Mantid::Geometry::IDetector*>::iterator it; - int pixnum=101191; - for (it=detectorgroup.begin();it!=detectorgroup.end();it++) - TS_ASSERT_EQUALS((*it)->getID(),pixnum++); - - // Test with spectra that does not exist - // Test that number of pixel=0 - TS_ASSERT_EQUALS(map->ndet(5),0); - // Test that trying to get the Detector throws. - boost::shared_ptr<Mantid::Geometry::IDetector> test; - TS_ASSERT_THROWS(test=map->getDetector(5),std::runtime_error); - */ - } -// void testWithManagedWorkspace() -// { -// ConfigService::Instance().updateConfig("UseManagedWS.properties"); -// //LoadRaw loader4; -// //loader4.initialize(); -// //loader4.setPropertyValue("Filename", inputFile); -// //loader4.setPropertyValue("OutputWorkspace", "managedws"); -// // TS_ASSERT_THROWS_NOTHING( loader4.execute() ) -// //TS_ASSERT( loader4.isExecuted() ) -// -// // Get back workspace and check it really is a ManagedWorkspace2D -// MatrixWorkspace_sptr output; -// TS_ASSERT_THROWS_NOTHING( output = AnalysisDataService::Instance().retrieve("managedws") ); -// TS_ASSERT( dynamic_cast<ManagedWorkspace2D*>(output.get()) ) -// } - - - void testTransvereDataset() - { - LoadMuonNexus nxL; - if ( !nxL.isInitialized() ) nxL.initialize(); - - // Now set required filename and output workspace name - std::string inputFile_musr00022725 = "MUSR00022725.nxs"; - nxL.setPropertyValue("FileName", inputFile_musr00022725); - - outputSpace="outermusr00022725"; - nxL.setPropertyValue("OutputWorkspace", outputSpace); - - TS_ASSERT_THROWS_NOTHING(nxL.execute()); - TS_ASSERT( nxL.isExecuted() ); - - // Test additional output parameters - std::string field = nxL.getProperty("MainFieldDirection"); - TS_ASSERT( field == "Transverse" ); - double timeZero = nxL.getProperty("TimeZero"); - TS_ASSERT_DELTA( timeZero, 0.55,0.001); - double firstgood = nxL.getProperty("FirstGoodData"); - TS_ASSERT_DELTA( firstgood, 0.656,0.001); - std::vector<double> deadTimes = nxL.getProperty("DeadTimes"); - TS_ASSERT_DELTA( deadTimes[0], 0.006,0.001); - TS_ASSERT_DELTA( deadTimes[deadTimes.size()-1], 0.011,0.001); - } - - void testExec2() - { - //test for multi period - // Now set required filename and output workspace name - inputFile2 = "emu00006475.nxs"; - nxLoad.setPropertyValue("FileName", inputFile2); - - outputSpace="outer2"; - nxLoad.setPropertyValue("OutputWorkspace", outputSpace); - nxLoad.setPropertyValue("EntryNumber", "1"); - int64_t entryNumber=nxLoad.getProperty("EntryNumber"); - - // - // Test execute to read file and populate workspace - // - TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); - TS_ASSERT( nxLoad.isExecuted() ); - // - // Test workspace data - should be 4 separate workspaces for this 4 period file - // - if(entryNumber==0) - { - WorkspaceGroup_sptr outGrp; - TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); - - } - //if entry number is given - if(entryNumber==1) - { - MatrixWorkspace_sptr output; - output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace); - - Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); - //Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); - // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; - TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); - // Check two X vectors are the same - TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); - // Check two Y arrays have the same number of elements - TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); - // Check that the time is as expected from bin boundary update - TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); - - // Check the unit has been set correctly - TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); - TS_ASSERT( ! output-> isDistribution() ); - - //check that sample name has been set correctly - //boost::shared_ptr<Sample> sample,sample2; - //sample = output->getSample(); - //sample2 = output2->getSample(); - //TS_ASSERT_EQUALS(sample->getName(), sample2->getName()); - TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); - - } - MatrixWorkspace_sptr output,output2,output3,output4; - WorkspaceGroup_sptr outGrp; - //if no entry number load the group workspace - if(entryNumber==0) - { - TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); - - (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_1")); - (output2 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_2")); - (output3 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_3")); - (output4 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_4")); - - Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); - Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); - // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; - TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); - // Check two X vectors are the same - TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); - // Check two Y arrays have the same number of elements - TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); - // Check one particular value - TS_ASSERT_EQUALS( output2D2->dataY(8)[502], 121); - // Check that the error on that value is correct - TS_ASSERT_EQUALS( output2D2->dataE(8)[502], 11); - // Check that the time is as expected from bin boundary update - TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); - - // Check the unit has been set correctly - TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); - TS_ASSERT( ! output-> isDistribution() ); - - //check that sample name has been set correctly - // boost::shared_ptr<Sample> sample,sample2; - //sample = output->getSample(); - // sample2 = output2->getSample(); - TS_ASSERT_EQUALS(output->sample().getName(), output2->sample().getName()); - TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); - - } - } - void testExec2withZeroEntryNumber() - { - //test for multi period - // Now set required filename and output workspace name - inputFile2 = "emu00006475.nxs"; - nxLoad.setPropertyValue("FileName", inputFile2); - - outputSpace="outer2"; - nxLoad.setPropertyValue("OutputWorkspace", outputSpace); - nxLoad.setPropertyValue("EntryNumber", "0"); - int64_t entryNumber=nxLoad.getProperty("EntryNumber"); - - // - // Test execute to read file and populate workspace - // - TS_ASSERT_THROWS_NOTHING(nxLoad.execute()); - TS_ASSERT( nxLoad.isExecuted() ); - // - // Test workspace data - should be 4 separate workspaces for this 4 period file - // - WorkspaceGroup_sptr outGrp; - TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); - - MatrixWorkspace_sptr output,output2,output3,output4; - //WorkspaceGroup_sptr outGrp; - //if no entry number load the group workspace - if(entryNumber==0) - { - //TS_ASSERT_THROWS_NOTHING(outGrp = AnalysisDataService::Instance().retrieveWS<WorkspaceGroup>(outputSpace)); - - (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_1")); - (output2 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_2")); - (output3 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_3")); - (output4 = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>(outputSpace+"_4")); - - Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); - Workspace2D_sptr output2D2 = boost::dynamic_pointer_cast<Workspace2D>(output2); - // Should be 32 for file inputFile = "../../../../Test/Nexus/emu00006475.nxs"; - TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 32); - // Check two X vectors are the same - TS_ASSERT( (output2D->dataX(3)) == (output2D->dataX(31)) ); - // Check two Y arrays have the same number of elements - TS_ASSERT_EQUALS( output2D->dataY(5).size(), output2D->dataY(17).size() ); - // Check one particular value - TS_ASSERT_EQUALS( output2D2->dataY(8)[502], 121); - // Check that the error on that value is correct - TS_ASSERT_EQUALS( output2D2->dataE(8)[502], 11); - // Check that the time is as expected from bin boundary update - TS_ASSERT_DELTA( output2D->dataX(11)[687], 10.738,0.001); - - // Check the unit has been set correctly - TS_ASSERT_EQUALS( output->getAxis(0)->unit()->unitID(), "Label" ); - TS_ASSERT( ! output-> isDistribution() ); - - //check that sample name has been set correctly - //boost::shared_ptr<Sample> sample,sample2; - // sample = output->getSample(); - // sample2 = output2->getSample(); - TS_ASSERT_EQUALS(output->sample().getName(), output2->sample().getName()); - TS_ASSERT_EQUALS(output->sample().getName(), "ptfe test"); - - } - } - - void testarrayin() - { - if ( !nxload3.isInitialized() ) nxload3.initialize(); - - nxload3.setPropertyValue("Filename", inputFile); - nxload3.setPropertyValue("OutputWorkspace", "outWS"); - nxload3.setPropertyValue("SpectrumList", "29,30,31"); - nxload3.setPropertyValue("SpectrumMin", "5"); - nxload3.setPropertyValue("SpectrumMax", "10"); - - TS_ASSERT_THROWS_NOTHING(nxload3.execute()); - TS_ASSERT( nxload3.isExecuted() ); - - // Get back the saved workspace - MatrixWorkspace_sptr output; - (output = AnalysisDataService::Instance().retrieveWS<MatrixWorkspace>("outWS")); - Workspace2D_sptr output2D = boost::dynamic_pointer_cast<Workspace2D>(output); - - // Should be 6 for selected input - TS_ASSERT_EQUALS( output2D->getNumberHistograms(), 9); - - // Check two X vectors are the same - TS_ASSERT( (output2D->dataX(1)) == (output2D->dataX(5)) ); - - // Check two Y arrays have the same number of elements - TS_ASSERT_EQUALS( output2D->dataY(2).size(), output2D->dataY(7).size() ); - - // Check one particular value - TS_ASSERT_EQUALS( output2D->dataY(8)[479], 144); - // Check that the error on that value is correct - TS_ASSERT_EQUALS( output2D->dataE(8)[479], 12); - // Check that the error on that value is correct - TS_ASSERT_DELTA( output2D->dataX(8)[479], 7.410, 0.0001); - } - -private: - LoadMuonNexus nxLoad,nxload2,nxload3; - std::string outputSpace; - std::string entryName; - std::string inputFile; - std::string inputFile2; - boost::shared_ptr<SpectraDetectorMap> map; }; + #endif /*LOADMUONNEXUSTEST_H_*/ diff --git a/Code/Mantid/Framework/Nexus/inc/MantidNexus/NexusClasses.h b/Code/Mantid/Framework/Nexus/inc/MantidNexus/NexusClasses.h index 1a215c19913..f63827b2aed 100644 --- a/Code/Mantid/Framework/Nexus/inc/MantidNexus/NexusClasses.h +++ b/Code/Mantid/Framework/Nexus/inc/MantidNexus/NexusClasses.h @@ -399,7 +399,7 @@ namespace Mantid alloc(n); getSlab(m_data.get(),start,m_size); } - protected: + private: /** Allocates memory for the data buffer * @param n :: The number of elements to allocate. */ @@ -559,7 +559,10 @@ namespace Mantid * @return NXInfo::stat is set to NX_ERROR if the dataset does not exist */ NXInfo getDataSetInfo(const std::string& name)const; - void close();///< Close this class + /// Returns whether an individual dataset is present + bool containsDataSet(const std::string & query) const; + /// Close this class + void close(); /// Opens this NXClass using NXopengrouppath. Can be slow (or is slow) void open(); /// Opens this NXClass using NXopengroup. It is fast, but the parent of this class must be open at diff --git a/Code/Mantid/Framework/Nexus/src/NexusClasses.cpp b/Code/Mantid/Framework/Nexus/src/NexusClasses.cpp index ab7cf50613d..153924ee68d 100644 --- a/Code/Mantid/Framework/Nexus/src/NexusClasses.cpp +++ b/Code/Mantid/Framework/Nexus/src/NexusClasses.cpp @@ -307,6 +307,11 @@ bool NXClass::containsGroup(const std::string & query) const return false; } +/** + * Returns NXInfo for a dataset + * @param name :: The name of the dataset + * @return NXInfo::stat is set to NX_ERROR if the dataset does not exist + */ NXInfo NXClass::getDataSetInfo(const std::string& name)const { NXInfo info; @@ -318,6 +323,14 @@ NXInfo NXClass::getDataSetInfo(const std::string& name)const return info; } +/** + * Returns whether an individual dataset is present. + */ +bool NXClass::containsDataSet(const std::string & query) const +{ + return getDataSetInfo( query ).stat != NX_ERROR; +} + //--------------------------------------------------------- // NXNote methods //--------------------------------------------------------- diff --git a/Test/AutoTestData/deltat_tdc_gpd_0900.nxs b/Test/AutoTestData/deltat_tdc_gpd_0900.nxs new file mode 100644 index 0000000000000000000000000000000000000000..7ca142986212bcd165779924d1a96ceb243176c1 GIT binary patch literal 75276 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 JKKlLq{{c_v>w^FQ literal 0 HcmV?d00001 -- GitLab