Newer
Older
Janik Zikovsky
committed
//----------------------------------------------------------------------
// Includes
//----------------------------------------------------------------------
#include "MantidDataHandling/LoadLogsFromSNSNexus.h"
Gigg, Martyn Anthony
committed
#include "MantidGeometry/Instrument/Instrument.h"
Janik Zikovsky
committed
#include "MantidKernel/ConfigService.h"
Janik Zikovsky
committed
#include "MantidKernel/DateAndTime.h"
Janik Zikovsky
committed
#include "MantidKernel/Timer.h"
Janik Zikovsky
committed
#include "MantidAPI/FileProperty.h"
#include <fstream>
#include <sstream>
#include <boost/algorithm/string/replace.hpp>
using std::cout;
using std::endl;
using std::map;
using std::string;
using std::vector;
Janik Zikovsky
committed
static bool VERBOSE = false;
Janik Zikovsky
committed
namespace Mantid
{
namespace DataHandling
Janik Zikovsky
committed
{
DECLARE_ALGORITHM(LoadLogsFromSNSNexus)
Janik Zikovsky
committed
/// Sets documentation strings for this algorithm
void LoadLogsFromSNSNexus::initDocs()
{
this->setWikiSummary("Loads sample logs (temperature, pulse charges, etc.) from a SNS NeXus file and adds it to the run information in a [[workspace]]. This is useful when using [[LoadEventPreNeXus]], to add sample logs after loading. ");
this->setOptionalMessage("Loads sample logs (temperature, pulse charges, etc.) from a SNS NeXus file and adds it to the run information in a workspace. This is useful when using LoadEventPreNeXus, to add sample logs after loading.");
}
Janik Zikovsky
committed
using namespace Kernel;
using namespace API;
Gigg, Martyn Anthony
committed
using Geometry::Instrument;
Janik Zikovsky
committed
/// Empty default constructor
LoadLogsFromSNSNexus::LoadLogsFromSNSNexus()
Gigg, Martyn Anthony
committed
{
Gigg, Martyn Anthony
committed
useAlgorithm("LoadNexusLogs");
Gigg, Martyn Anthony
committed
deprecatedDate("2011-06-01");
}
Janik Zikovsky
committed
/// Initialisation method.
void LoadLogsFromSNSNexus::init()
{
// When used as a sub-algorithm the workspace name is not used - hence the "Anonymous" to satisfy the validator
declareProperty(
Janik Zikovsky
committed
new WorkspaceProperty<MatrixWorkspace>("Workspace","Anonymous",Direction::InOut),
Janik Zikovsky
committed
"The name of the workspace in which to import the sample logs." );
Janik Zikovsky
committed
declareProperty(new FileProperty("Filename", "", FileProperty::Load, ".nxs"),
Gigg, Martyn Anthony
committed
"The name (including its full or relative path) of the Nexus file to\n"
"attempt to load the instrument from. The file extension must either be\n"
".nxs or .NXS" );
declareProperty(new PropertyWithValue<bool>("OverwriteLogs", true, Direction::Input));
Janik Zikovsky
committed
}
/** Executes the algorithm. Reading in the file and creating and populating
* the output workspace
*
*/
void LoadLogsFromSNSNexus::exec()
{
// Retrieve the filename from the properties
m_filename = getPropertyValue("Filename");
// Get the input workspace
Janik Zikovsky
committed
WS = getProperty("Workspace");
// top level file information
Janik Zikovsky
committed
::NeXus::File file(m_filename);
Janik Zikovsky
committed
g_log.information() << "NeXus file found: " << file.inquireFile() << endl;
//Start with the base entry
file.openGroup("entry", "NXentry");
//Now go to the DAS logs
file.openGroup("DASlogs", "NXgroup");
// print out the entry level fields
map<string, string> entries = file.getEntries();
Janik Zikovsky
committed
::NeXus::Info info;
Janik Zikovsky
committed
map<string,string>::const_iterator it = entries.begin();
for (; it != entries.end(); it++)
{
std::string entry_name(it->first);
std::string entry_class(it->second);
if ((entry_class == "NXlog") ||(entry_class == "NXpositioner"))
Janik Zikovsky
committed
{
loadSampleLog(file, entry_name, entry_class);
}
}
file.closeGroup();
Janik Zikovsky
committed
Janik Zikovsky
committed
try
{
//Use the DAS logs to integrate the proton charge (if any).
WS->mutableRun().integrateProtonCharge();
}
catch (Exception::NotFoundError &)
Janik Zikovsky
committed
{
//Ignore not found property error.
}
Janik Zikovsky
committed
return;
}
/** Loads an entry from a previously-open NXS file as a log entry
* in the workspace's run.
Janik Zikovsky
committed
*
* @param file: NXS file handle. MUST BE PASSED BY REFERENCE otherwise there
* occurs a segfault.
* @param entry_name, entry_class: name and class of NXlog to open.
Janik Zikovsky
committed
void LoadLogsFromSNSNexus::loadSampleLog(::NeXus::File& file, std::string entry_name, std::string entry_class)
Janik Zikovsky
committed
{
// whether or not to overwrite logs on workspace
bool overwritelogs = this->getProperty("OverwriteLogs");
Janik Zikovsky
committed
file.openGroup(entry_name, entry_class);
// Validate the NX log class.
map<string, string> entries = file.getEntries();
if ((entries.find("value") == entries.end()) ||
(entries.find("time") == entries.end()) )
{
Janik Zikovsky
committed
g_log.warning() << "Invalid NXlog entry " << entry_name << " found. Did not contain 'value' and 'time'.\n";
file.closeGroup();
Janik Zikovsky
committed
return;
}
Janik Zikovsky
committed
::NeXus::Info info;
Janik Zikovsky
committed
//Two possible types of properties:
vector<double> values;
vector<int> values_int;
bool isTimeSeries = false;
bool isInt = false;
file.openData("value");
Janik Zikovsky
committed
//Get the units of the property
std::string units("");
try
{
file.getAttr("units", units);
}
catch (::NeXus::Exception &)
Janik Zikovsky
committed
{
//Ignore missing units field.
units = "";
}
Janik Zikovsky
committed
//If there is more than one entry, it is a timeseries
Janik Zikovsky
committed
info = file.getInfo();
//isTimeSeries = (info.dims[0] > 1);
isTimeSeries = true;
Janik Zikovsky
committed
Janik Zikovsky
committed
Timer timer1;
Janik Zikovsky
committed
try
Janik Zikovsky
committed
{
Janik Zikovsky
committed
//Get the data (convert types if necessary)
if (file.isDataInt())
Janik Zikovsky
committed
{
Janik Zikovsky
committed
isInt = true;
file.getDataCoerce(values_int);
// if (values_int.size() == 1)
// {
// WS->mutableRun().addProperty(entry_name, values_int[0], units);
// }
Janik Zikovsky
committed
Janik Zikovsky
committed
}
Janik Zikovsky
committed
else
Janik Zikovsky
committed
{
//Try to get as doubles.
file.getDataCoerce(values);
// if (values.size() == 1)
// {
// WS->mutableRun().addProperty(entry_name, values[0], units);
// }
Janik Zikovsky
committed
}
}
catch (::NeXus::Exception &e)
Janik Zikovsky
committed
{
g_log.warning() << "NXlog entry " << entry_name << " gave an error when loading 'value' data:'" << e.what() << "'.\n";
file.closeData();
file.closeGroup();
return;
}
Janik Zikovsky
committed
if (VERBOSE) std::cout << "getDataCoerce took " << timer1.elapsed() << " sec.\n";
Janik Zikovsky
committed
Janik Zikovsky
committed
file.closeData();
if (isTimeSeries)
{
// --- Time series property ---
//Get the times
vector<double> time_double;
Janik Zikovsky
committed
vector<DateAndTime> times;
Janik Zikovsky
committed
Janik Zikovsky
committed
try {
file.openData("time");
}
catch (::NeXus::Exception &e)
Janik Zikovsky
committed
{
g_log.warning() << "NXlog entry " << entry_name << " gave an error when opening the time field '" << e.what() << "'.\n";
file.closeGroup();
return;
}
Janik Zikovsky
committed
Janik Zikovsky
committed
//----- Start time is an ISO8601 string date and time. ------
Janik Zikovsky
committed
std::string start;
Janik Zikovsky
committed
try {
file.getAttr("start", start);
}
catch (::NeXus::Exception &)
Janik Zikovsky
committed
{
//Some logs have "offset" instead of start
try {
file.getAttr("offset", start);
}
catch (::NeXus::Exception &)
Janik Zikovsky
committed
{
g_log.warning() << "NXlog entry " << entry_name << " has no start time indicated.\n";
file.closeData();
file.closeGroup();
return;
}
}
Janik Zikovsky
committed
//Convert to date and time
Janik Zikovsky
committed
Kernel::DateAndTime start_time = Kernel::DateAndTime(start);
Janik Zikovsky
committed
std::string time_units;
file.getAttr("units", time_units);
if (time_units != "second")
{
g_log.warning() << "NXlog entry " << entry_name << " has time units of '" << time_units << "', which are unsupported. 'second' is the only supported time unit.\n";
file.closeData();
file.closeGroup();
return;
}
Janik Zikovsky
committed
Timer timer2;
Janik Zikovsky
committed
//--- Load the seconds into a double array ---
try {
file.getDataCoerce(time_double);
}
catch (::NeXus::Exception &e)
Janik Zikovsky
committed
{
g_log.warning() << "NXlog entry " << entry_name << "'s time field could not be loaded: '" << e.what() << "'.\n";
file.closeData();
file.closeGroup();
return;
}
Janik Zikovsky
committed
file.closeData();
Janik Zikovsky
committed
if (VERBOSE) std::cout << "getDataCoerce for the seconds field took " << timer2.elapsed() << " sec.\n";
Janik Zikovsky
committed
Janik Zikovsky
committed
if (isInt)
{
//Make an int TSP
TimeSeriesProperty<int> * tsp = new TimeSeriesProperty<int>(entry_name);
tsp->create(start_time, time_double, values_int);
Janik Zikovsky
committed
tsp->setUnits( units );
WS->mutableRun().addProperty( tsp, overwritelogs );
Janik Zikovsky
committed
}
else
{
//Make a double TSP
TimeSeriesProperty<double> * tsp = new TimeSeriesProperty<double>(entry_name);
Janik Zikovsky
committed
Timer timer3;
Janik Zikovsky
committed
tsp->create(start_time, time_double, values);
Janik Zikovsky
committed
if (VERBOSE) std::cout << "creating a TSP took " << timer3.elapsed() << " sec.\n";
Janik Zikovsky
committed
tsp->setUnits( units );
WS->mutableRun().addProperty( tsp, overwritelogs );
// Trick to free memory?
std::vector<double>().swap(time_double);
std::vector<double>().swap(values);
Janik Zikovsky
committed
Janik Zikovsky
committed
}
}
file.closeGroup();
}
} // namespace DataHandling
Janik Zikovsky
committed
} // namespace Mantid