-
Peterson, Peter authored
Also redo Timer.cpp and XMLFormatter.h to use c++11 constructs.
Peterson, Peter authoredAlso redo Timer.cpp and XMLFormatter.h to use c++11 constructs.
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
DateValidator.cpp 3.39 KiB
#include "MantidKernel/DateValidator.h"
#include <boost/lexical_cast.hpp>
#include <boost/make_shared.hpp>
#include <ctime>
namespace Mantid {
namespace Kernel {
namespace {
//
// Keeps time.h out of the header
//
/** Checks the validity of date string ,expected format is "DD/MM/YYYY"
* @param sDate :: expected date format
* @param error :: string which describes the error
* @return tm structure with the date information filled in
*/
struct tm getTimeValue(const std::string &sDate, std::string &error) {
struct tm timeinfo;
// Zero it out
timeinfo.tm_sec = 0;
timeinfo.tm_min = 0;
timeinfo.tm_hour = 0;
timeinfo.tm_mday = 0;
timeinfo.tm_mon = 0;
timeinfo.tm_year = 0;
timeinfo.tm_wday = 0;
timeinfo.tm_yday = 0;
timeinfo.tm_isdst = -1;
#ifndef _WIN32
timeinfo.tm_gmtoff = 0;
timeinfo.tm_zone = nullptr;
#endif
std::basic_string<char>::size_type index, off = 0;
int day, month, year;
// look for the first '/' to extract day part from the date string
index = sDate.find('/', off);
if (index == std::string::npos) {
error = "Invalid Date:date format must be DD/MM/YYYY";
timeinfo.tm_mday = 0;
return timeinfo;
}
// get day part of the date
try {
day = boost::lexical_cast<int>(sDate.substr(off, index - off).c_str());
} catch (boost::bad_lexical_cast &) {
error = "Invalid Date";
return timeinfo;
}
timeinfo.tm_mday = day;
// change the offset to the next position after "/"
off = index + 1;
// look for 2nd '/' to get month part from the date string
index = sDate.find('/', off);
if (index == std::string::npos) {
error = "Invalid Date:date format must be DD/MM/YYYY";
return timeinfo;
}
// now get the month part
try {
month = boost::lexical_cast<int>(sDate.substr(off, index - off).c_str());
} catch (boost::bad_lexical_cast &) {
error = "Invalid Date";
return timeinfo;
}
timeinfo.tm_mon = month - 1;
// change the offset to the position after "/"
off = index + 1;
// now get the year part from the date string
try {
year = boost::lexical_cast<int>(sDate.substr(off, 4).c_str());
} catch (boost::bad_lexical_cast &) {
error = "Invalid Date";
return timeinfo;
}
timeinfo.tm_year = year - 1900;
return timeinfo;
}
}
/// create a copy of the the validator
IValidator_sptr DateValidator::clone() const {
return boost::make_shared<DateValidator>(*this);
}
/** Checks the given value is a valid date
* @param value :: input date property to validate
* @return a string which describes the error else ""
*/
std::string DateValidator::checkValidity(const std::string &value) const {
// empty strings are allowed
if (value.empty()) {
return "";
}
std::string formaterror;
struct tm timeinfo = getTimeValue(value, formaterror);
if (!formaterror.empty()) {
return formaterror;
}
if (timeinfo.tm_mday < 1 || timeinfo.tm_mday > 31) {
return "Invalid Date:Day part of the Date parameter must be between 1 and "
"31";
}
if (timeinfo.tm_mon < 0 || timeinfo.tm_mon > 11) {
return "Invalid Date:Month part of the Date parameter must be between 1 "
"and 12";
}
// get current time
time_t rawtime;
time(&rawtime);
struct tm *currenttime = localtime(&rawtime);
if (timeinfo.tm_year > currenttime->tm_year) {
return "Invalid Date:Year part of the Date parameter can not be greater "
"than the current year";
}
return "";
}
}
}