Newer
Older
#include "MantidAlgorithms/ChangeTimeZero.h"
#include "MantidAPI/IEventWorkspace.h"
#include "MantidAPI/Run.h"
#include "MantidAlgorithms/ChangePulsetime.h"
#include "MantidAlgorithms/CloneWorkspace.h"
#include "MantidDataObjects/EventList.h"
#include "MantidKernel/ArrayProperty.h"
#include "MantidKernel/DateAndTime.h"
#include "MantidKernel/DateTimeValidator.h"
#include "MantidKernel/PropertyWithValue.h"
#include "MantidKernel/System.h"
#include "MantidKernel/TimeSeriesProperty.h"
namespace Mantid {
namespace Algorithms {
// Register the algorithm into the AlgorithmFactory
DECLARE_ALGORITHM(ChangeTimeZero)
using namespace Mantid::Kernel;
using namespace Mantid::API;
using Types::Core::DateAndTime;
namespace {
/**
* General check if we are dealing with a time series
* @param prop :: the property which is being checked
* @return True if the proerpty is a time series, otherwise false.
*/
bool isTimeSeries(Mantid::Kernel::Property *prop) {
auto isTimeSeries = false;
if (dynamic_cast<Mantid::Kernel::ITimeSeriesProperty *>(prop)) {
isTimeSeries = true;
}
return isTimeSeries;
}
/** Initialize the algorithm's properties.
*/
void ChangeTimeZero::init() {
declareProperty(make_unique<WorkspaceProperty<MatrixWorkspace>>(
"InputWorkspace", "", Direction::Input),
declareProperty<double>("RelativeTimeOffset", m_defaultTimeShift,
Anton Piccardo-Selg
committed
"A relative time offset in seconds.");
declareProperty("AbsoluteTimeOffset", m_defaultAbsoluteTimeShift,
Anton Piccardo-Selg
committed
"An absolute time offset as an ISO8601 string "
"(YYYY-MM-DDTHH:MM::SS, eg 2013-10-25T13:58:03).");
declareProperty(make_unique<WorkspaceProperty<MatrixWorkspace>>(
"OutputWorkspace", "", Direction::Output),
"An output workspace.");
}
/** Execute the algorithm.
*/
void ChangeTimeZero::exec() {
MatrixWorkspace_sptr in_ws = getProperty("InputWorkspace");
// Create a new target workspace if it does not exist
Anton Piccardo-Selg
committed
const double progressStartCreateOutputWs = 0.0;
const double progressStopCreateOutputWs = 0.3;
Anton Piccardo-Selg
committed
MatrixWorkspace_sptr out_ws = createOutputWS(
in_ws, progressStartCreateOutputWs, progressStopCreateOutputWs);
auto timeShift = getTimeShift(out_ws);
Anton Piccardo-Selg
committed
// Set up remaining progress points
const double progressStartShiftTimeLogs = progressStopCreateOutputWs;
double progressStopShiftTimeLogs = progressStartShiftTimeLogs;
if (boost::dynamic_pointer_cast<Mantid::API::IEventWorkspace>(out_ws)) {
progressStopShiftTimeLogs = progressStartShiftTimeLogs + 0.1;
} else {
progressStopShiftTimeLogs = 1.0;
}
const double progressStartShiftNeutrons = progressStopShiftTimeLogs;
const double progressStopShiftNeutrons = 1.0;
Anton Piccardo-Selg
committed
// Initialize progress reporting.
shiftTimeOfLogs(out_ws, timeShift, progressStartShiftTimeLogs,
progressStopShiftTimeLogs);
// Change the time stamps on the neutrons
Anton Piccardo-Selg
committed
shiftTimeOfNeutrons(out_ws, timeShift, progressStartShiftNeutrons,
progressStopShiftNeutrons);
setProperty("OutputWorkspace", out_ws);
}
/**
* Create a new output workspace if required
* @param input :: pointer to an input workspace
Anton Piccardo-Selg
committed
* @param startProgress :: start point of the progress
* @param stopProgress :: end point of the progress
* @returns :: pointer to the outputworkspace
*/
API::MatrixWorkspace_sptr
Anton Piccardo-Selg
committed
ChangeTimeZero::createOutputWS(API::MatrixWorkspace_sptr input,
double startProgress, double stopProgress) {
MatrixWorkspace_sptr output = getProperty("OutputWorkspace");
// Check whether input == output to see whether a new workspace is required.
Anton Piccardo-Selg
committed
IAlgorithm_sptr duplicate =
createChildAlgorithm("CloneWorkspace", startProgress, stopProgress);
duplicate->initialize();
duplicate->setProperty<API::Workspace_sptr>(
"InputWorkspace", boost::dynamic_pointer_cast<API::Workspace>(input));
duplicate->execute();
API::Workspace_sptr temp = duplicate->getProperty("OutputWorkspace");
output = boost::dynamic_pointer_cast<API::MatrixWorkspace>(temp);
* Get the time shift that was specified by the user. If the time is
* absolute, we need to convert it to relative time.
* @param ws :: a workspace with time stamp information
* @returns A time shift in seconds
*/
double ChangeTimeZero::getTimeShift(API::MatrixWorkspace_sptr ws) const {
auto timeShift = m_defaultTimeShift;
// Check if we are dealing with an absolute time
Anton Piccardo-Selg
committed
std::string timeOffset = getProperty("AbsoluteTimeOffset");
if (isAbsoluteTimeShift(timeOffset)) {
DateAndTime desiredTime(timeOffset);
DateAndTime originalTime(getStartTimeFromWorkspace(ws));
timeShift = DateAndTime::secondsFromDuration(desiredTime - originalTime);
timeShift = getProperty("RelativeTimeOffset");
}
return timeShift;
}
/**
* @param ws :: a workspace
* @param timeShift :: the time shift that is applied to the log files
Anton Piccardo-Selg
committed
* @param startProgress :: start point of the progress
* @param stopProgress :: end point of the progress
*/
void ChangeTimeZero::shiftTimeOfLogs(Mantid::API::MatrixWorkspace_sptr ws,
Anton Piccardo-Selg
committed
double timeShift, double startProgress,
double stopProgress) {
// We need to change the entries for each log which can be:
// 1. any time series: here we change the time values
// 2. string properties: here we change the values if they are ISO8601 times
Anton Piccardo-Selg
committed
auto logs = ws->mutableRun().getLogData();
Progress prog(this, startProgress, stopProgress, logs.size());
for (auto &log : logs) {
if (isTimeSeries(log)) {
shiftTimeInLogForTimeSeries(ws, log, timeShift);
dynamic_cast<PropertyWithValue<std::string> *>(log)) {
shiftTimeOfLogForStringProperty(stringProperty, timeShift);
Anton Piccardo-Selg
committed
prog.report(name());
/**
* Shift the time in a time series. This is similar to the implementation in
* @param ws :: a matrix workspace
* @param prop :: a time series log
* @param timeShift :: the time shift in seconds
*/
void ChangeTimeZero::shiftTimeInLogForTimeSeries(
Mantid::API::MatrixWorkspace_sptr ws, Mantid::Kernel::Property *prop,
Anton Piccardo-Selg
committed
double timeShift) const {
if (auto timeSeries =
dynamic_cast<Mantid::Kernel::ITimeSeriesProperty *>(prop)) {
auto newlog = timeSeries->cloneWithTimeShift(timeShift);
ws->mutableRun().addProperty(newlog, true);
/**
* Shift the times in a log of a string property
* @param logEntry :: the string property
* @param timeShift :: the time shift.
*/
void ChangeTimeZero::shiftTimeOfLogForStringProperty(
Anton Piccardo-Selg
committed
PropertyWithValue<std::string> *logEntry, double timeShift) const {
// Parse the log entry and replace all ISO8601 strings with an adjusted value
if (checkForDateTime(value)) {
DateAndTime dateTime(value);
DateAndTime shiftedTime = dateTime + timeShift;
logEntry->setValue(shiftedTime.toISO8601String());
}
}
/**
* Shift the time of the neutrons
* @param ws :: a matrix workspace
* @param timeShift :: the time shift in seconds
Anton Piccardo-Selg
committed
* @param startProgress :: start point of the progress
* @param stopProgress :: end point of the progress
*/
void ChangeTimeZero::shiftTimeOfNeutrons(Mantid::API::MatrixWorkspace_sptr ws,
Anton Piccardo-Selg
committed
double timeShift, double startProgress,
double stopProgress) {
if (auto eventWs =
boost::dynamic_pointer_cast<Mantid::API::IEventWorkspace>(ws)) {
// Use the change pulse time algorithm to change the neutron time stamp
auto alg =
createChildAlgorithm("ChangePulsetime", startProgress, stopProgress);
alg->initialize();
alg->setProperty("InputWorkspace", eventWs);
alg->setProperty("OutputWorkspace", eventWs);
alg->setProperty("TimeOffset", timeShift);
alg->execute();
}
/**
* Extract the first good frame of a workspace
* @param ws :: a workspace
* @returns the date and time of the first good frame
DateAndTime
ChangeTimeZero::getStartTimeFromWorkspace(API::MatrixWorkspace_sptr ws) const {
auto run = ws->run();
// Check for the first good frame in the log
Mantid::Kernel::TimeSeriesProperty<double> *goodFrame = nullptr;
try {
goodFrame = run.getTimeSeriesProperty<double>("proton_charge");
} catch (const std::invalid_argument &) {
throw std::invalid_argument("ChangeTimeZero: The log needs a proton_charge "
"time series to determine the zero time.");
}
if (goodFrame->size() > 0) {
startTime = goodFrame->firstTime();
}
return startTime;
}
/**
* Check the inputs for invalid values
* @returns A map with validation warnings.
*/
std::map<std::string, std::string> ChangeTimeZero::validateInputs() {
std::map<std::string, std::string> invalidProperties;
// Check the time offset for either a value or a date time
double relativeTimeOffset = getProperty("RelativeTimeOffset");
std::string absoluteTimeOffset = getProperty("AbsoluteTimeOffset");
Anton Piccardo-Selg
committed
auto isRelative = isRelativeTimeShift(relativeTimeOffset);
auto absoluteTimeInput = absoluteTimeOffset != m_defaultAbsoluteTimeShift;
Anton Piccardo-Selg
committed
auto isAbsolute = isAbsoluteTimeShift(absoluteTimeOffset);
// If both inputs are being used, then return straight away.
Anton Piccardo-Selg
committed
if (isRelative && absoluteTimeInput) {
invalidProperties.emplace("RelativeTimeOffset",
"You can either specify a relative time shift or "
"an absolute time shift.");
invalidProperties.emplace("AbsoluteTimeOffset",
"You can either specify a relative time shift or "
return invalidProperties;
Anton Piccardo-Selg
committed
} else if (!isRelative && !isAbsolute) {
Anton Piccardo-Selg
committed
"RelativeTimeOffset",
"TimeOffset must either be a numeric "
"value or a ISO8601 (YYYY-MM-DDTHH:MM::SS) date-time stamp.");
invalidProperties.emplace(
Anton Piccardo-Selg
committed
"AbsoluteTimeOffset",
"TimeOffset must either be a numeric "
"value or a ISO8601 (YYYY-MM-DDTHH:MM::SS) date-time stamp.");
// If we are dealing with an absolute time we need to ensure that the
// proton_charge entry exists
Anton Piccardo-Selg
committed
if (isAbsolute) {
MatrixWorkspace_sptr ws = getProperty("InputWorkspace");
if (ws) {
auto run = ws->run();
try {
run.getTimeSeriesProperty<double>("proton_charge");
} catch (...) {
"InputWorkspace",
"A TimeOffset with an absolute time requires the "
"input workspace to have a proton_charge property in "
/** Can the string be transformed to double
* @param val :: value to check
* @return True if the string can be cast to double and otherwise false.
*/
Anton Piccardo-Selg
committed
bool ChangeTimeZero::checkForDouble(std::string val) const {
try {
boost::lexical_cast<double>(val);
isDouble = true;
} catch (boost::bad_lexical_cast const &) {
}
return isDouble;
}
/** Can the string be transformed to a DateTime object
* @param val :: value to check
* @return True if the string can be cast to a DateTime object and otherwise
* false.
*/
Anton Piccardo-Selg
committed
bool ChangeTimeZero::checkForDateTime(const std::string &val) const {
// Hedge for bad lexical casts in the DateTimeValidator
try {
Anton Piccardo-Selg
committed
DateTimeValidator validator = DateTimeValidator();
isDateTime = validator.isValid(val).empty();
} catch (...) {
isDateTime = false;
}
return isDateTime;
}
Anton Piccardo-Selg
committed
/**
* Checks if a relative offset has been set
* @param offset :: the offset
* @returns true if the offset has been set
*/
bool ChangeTimeZero::isRelativeTimeShift(double offset) const {
return offset != m_defaultTimeShift;
Anton Piccardo-Selg
committed
}
/**
* Checks if an absolute offset has been set
* @param offset :: the offset
* @returns true if the offset has been set
*/
bool ChangeTimeZero::isAbsoluteTimeShift(const std::string &offset) const {
return offset != m_defaultAbsoluteTimeShift && checkForDateTime(offset);
Anton Piccardo-Selg
committed
}
} // namespace Algorithms