Newer
Older
Roman Tolchenov
committed
//----------------------------------------------------------------------
// Includes
//----------------------------------------------------------------------
#include <algorithm>
Roman Tolchenov
committed
#include "MantidKernel/FacilityInfo.h"
#include "MantidKernel/ConfigService.h"
#include "MantidKernel/Exception.h"
#include "MantidKernel/Logger.h"
#include "MantidKernel/Strings.h"
Roman Tolchenov
committed
#include <boost/algorithm/string.hpp>
#include <Poco/DOM/Element.h>
#include <Poco/DOM/NodeList.h>
#include <Poco/StringTokenizer.h>
Roman Tolchenov
committed
using Poco::XML::Element;
namespace Mantid {
namespace Kernel {
Roman Tolchenov
committed
namespace {
/// static logger
Logger g_log("FacilityInfo");
Roman Tolchenov
committed
/** Constructor.
Janik Zikovsky
committed
* @param elem :: The Poco::XML::Element to read the data from
* @throw std::runtime_error if name or file extensions are not defined
Roman Tolchenov
committed
*/
FacilityInfo::FacilityInfo(const Poco::XML::Element *elem)
: m_catalogs(elem), m_name(elem->getAttribute("name")), m_zeroPadding(0),
m_delimiter(), m_extensions(), m_archiveSearch(), m_instruments(),
m_liveListener(), m_computeResources() {
if (m_name.empty()) {
g_log.error("Facility name is not defined");
throw std::runtime_error("Facility name is not defined");
Roman Tolchenov
committed
}
// Fill the various fields from the XML
fillZeroPadding(elem);
fillDelimiter(elem);
fillExtensions(elem);
fillArchiveNames(elem);
fillLiveListener(elem);
fillInstruments(elem); // Make sure this is last as it picks up some defaults
// that are set above
}
/// Called from constructor to fill zero padding field
void FacilityInfo::fillZeroPadding(const Poco::XML::Element *elem) {
Roman Tolchenov
committed
std::string paddingStr = elem->getAttribute("zeropadding");
if (paddingStr.empty() ||
!Mantid::Kernel::Strings::convert(paddingStr, m_zeroPadding)) {
Roman Tolchenov
committed
m_zeroPadding = 0;
}
Roman Tolchenov
committed
/// Called from constructor to fill default delimiter
void FacilityInfo::fillDelimiter(const Poco::XML::Element *elem) {
Campbell, Stuart
committed
// The string to separate the instrument name and the run number.
m_delimiter = elem->getAttribute("delimiter");
Campbell, Stuart
committed
/// Called from constructor to fill file extensions
void FacilityInfo::fillExtensions(const Poco::XML::Element *elem) {
Roman Tolchenov
committed
std::string extsStr = elem->getAttribute("FileExtensions");
Roman Tolchenov
committed
g_log.error("No file extensions defined");
throw std::runtime_error("No file extensions defined");
}
typedef Poco::StringTokenizer tokenizer;
tokenizer exts(extsStr, ",",
tokenizer::TOK_IGNORE_EMPTY | tokenizer::TOK_TRIM);
for (tokenizer::Iterator it = exts.begin(); it != exts.end(); ++it) {
Roman Tolchenov
committed
addExtension(*it);
}
Roman Tolchenov
committed
/**
* Add new extension. Adds both a lowercase and uppercase version
* @param ext :: File extension, including the dot, e.g. ".nxs" or ".raw"
*/
void FacilityInfo::addExtension(const std::string &ext) {
auto it = std::find(m_extensions.begin(), m_extensions.end(), ext);
if (it == m_extensions.end())
m_extensions.push_back(ext);
}
/// Called from constructor to fill archive interface names
void FacilityInfo::fillArchiveNames(const Poco::XML::Element *elem) {
Poco::AutoPtr<Poco::XML::NodeList> pNL_archives =
elem->getElementsByTagName("archive");
if (pNL_archives->length() > 1) {
Roman Tolchenov
committed
g_log.error("Facility must have only one archive tag");
throw std::runtime_error("Facility must have only one archive tag");
} else if (pNL_archives->length() == 1) {
Poco::AutoPtr<Poco::XML::NodeList> pNL_interfaces =
elem->getElementsByTagName("archiveSearch");
for (unsigned int i = 0; i < pNL_interfaces->length(); ++i) {
Poco::XML::Element *elem =
dynamic_cast<Poco::XML::Element *>(pNL_interfaces->item(i));
Roman Tolchenov
committed
std::string plugin = elem->getAttribute("plugin");
m_archiveSearch.push_back(plugin);
Roman Tolchenov
committed
}
}
}
Roman Tolchenov
committed
/// Called from constructor to fill instrument list
void FacilityInfo::fillInstruments(const Poco::XML::Element *elem) {
Poco::AutoPtr<Poco::XML::NodeList> pNL_instrument =
elem->getElementsByTagName("instrument");
unsigned long n = pNL_instrument->length();
m_instruments.reserve(n);
Roman Tolchenov
committed
for (unsigned long i = 0; i < n; ++i) {
Poco::XML::Element *elem =
dynamic_cast<Poco::XML::Element *>(pNL_instrument->item(i));
if (elem) {
try {
InstrumentInfo instr(this, elem);
Roman Tolchenov
committed
m_instruments.push_back(instr);
} catch (...) { /*skip this instrument*/
Roman Tolchenov
committed
}
}
}
if (m_instruments.empty()) {
throw std::runtime_error("Facility " + m_name +
" does not have any instruments;");
Roman Tolchenov
committed
}
/// Called from constructor to fill live listener name
void FacilityInfo::fillLiveListener(const Poco::XML::Element *elem) {
// Get the first livedata element (will be NULL if there's none)
Element *live = elem->getChildElement("livedata");
if (live) {
// Get the name of the listener - empty string will be returned if missing
m_liveListener = live->getAttribute("listener");
}
}
/// Called from constructor to fill compute resources map
void FacilityInfo::fillComputeResources(const Poco::XML::Element *elem) {
Poco::AutoPtr<Poco::XML::NodeList> pNL_compute =
elem->getElementsByTagName("computeResource");
unsigned long n = pNL_compute->length();
for (unsigned long i = 0; i < n; i++) {
Poco::XML::Element *elem =
dynamic_cast<Poco::XML::Element *>(pNL_compute->item(i));
Federico Montesino Pouzols
committed
if (elem) {
try {
ComputeResourceInfo cr(this, elem);
m_computeResInfos.push_back(cr);
g_log.debug() << "Compute resource found: " << cr << std::endl;
} catch (...) { // next resource...
}
std::string name = elem->getAttribute("name");
// TODO: this is a bit of duplicate effort at the moment, until
// RemoteJobManager goes away from here (then this would be
// removed), see header for details.
m_computeResources.insert(std::make_pair(
name,
boost::shared_ptr<RemoteJobManager>(new RemoteJobManager(elem))));
}
Roman Tolchenov
committed
/**
* Returns instrument with given name
Roman Tolchenov
committed
* @param iName Instrument name
* @return the instrument information object
Janik Zikovsky
committed
* @throw NotFoundError if iName was not found
Roman Tolchenov
committed
*/
const InstrumentInfo &FacilityInfo::instrument(std::string iName) const {
if (iName.empty()) {
iName = ConfigService::Instance().getString("default.instrument");
g_log.debug() << "Blank instrument specified, using default instrument of "
<< iName << "." << std::endl;
if (iName.empty()) {
Roman Tolchenov
committed
std::vector<InstrumentInfo>::const_iterator it = m_instruments.begin();
for (; it != m_instruments.end(); ++it) {
if (boost::iequals(it->name(), iName)) // Case-insensitive search
g_log.debug() << "Instrument '" << iName << "' found as " << it->name()
<< " at " << name() << "." << std::endl;
return *it;
}
}
// if unsuccessful try shortname
for (it = m_instruments.begin(); it != m_instruments.end(); ++it) {
if (boost::iequals(it->shortName(), iName)) // Case-insensitive search
Roman Tolchenov
committed
{
g_log.debug() << "Instrument '" << iName << "' found as " << it->name()
<< " at " << name() << "." << std::endl;
Roman Tolchenov
committed
return *it;
}
}
g_log.debug("Instrument " + iName + " not found in facility " + name());
throw Exception::NotFoundError("FacilityInfo", iName);
Roman Tolchenov
committed
}
/**
Federico Montesino Pouzols
committed
* Get the vector of available compute resources
* @return vector of ComputeResourInfo for the current facility
Roman Tolchenov
committed
*/
Federico Montesino Pouzols
committed
std::vector<ComputeResourceInfo> FacilityInfo::computeResInfos() const {
return m_computeResInfos;
}
/**
* Returns a list of instruments of given technique
* @param tech :: Technique name
* @return a list of instrument information objects
*/
std::vector<InstrumentInfo>
FacilityInfo::instruments(const std::string &tech) const {
Roman Tolchenov
committed
std::vector<InstrumentInfo> out;
Federico Montesino Pouzols
committed
auto it = m_instruments.begin();
for (; it != m_instruments.end(); ++it) {
if (it->techniques().count(tech)) {
Roman Tolchenov
committed
out.push_back(*it);
}
}
return out;
}
Federico Montesino Pouzols
committed
* Returns a vector of the names of the available compute resources
* @return vector of strings of the compute resource names
*/
std::vector<std::string> FacilityInfo::computeResources() const {
std::vector<std::string> names;
ComputeResourcesMap::const_iterator it = m_computeResources.begin();
while (it != m_computeResources.end()) {
names.push_back((*it).first);
}
return names;
}
Federico Montesino Pouzols
committed
/**
* Get a compute resource by name
*
* @param name Name as specified in the facilities definition file
*
* @return the named compute resource
*
* @throws NotFoundError if the resource is not found/available.
*/
const ComputeResourceInfo &
FacilityInfo::computeResource(const std::string &name) const {
if (name.empty()) {
g_log.debug("Cannot find a compute resource without name "
"(empty).");
throw Exception::NotFoundError("FacilityInfo, empty compute resource name",
name);
}
auto it = m_computeResInfos.begin();
for (; it != m_computeResInfos.end(); ++it) {
if (it->name() == name) {
g_log.debug() << "Compute resource '" << name << "' found at facility "
<< this->name() << "." << std::endl;
return *it;
}
Federico Montesino Pouzols
committed
}
g_log.debug() << "Could not find requested compute resource: " << name
<< " in facility " << this->name() << "." << std::endl;
throw Exception::NotFoundError(
"FacilityInfo, missing compute resource, it does not seem to be defined "
"in the facility '" +
this->name() + "' - ",
name);
Federico Montesino Pouzols
committed
}
/**
* Returns a reference to the requested remote job manager
* @param name :: Name of the cluster we want to submit jobs to
* @return a shared pointer to the RemoteJobManager instance (or
* Null if the name wasn't recognized)
*/
boost::shared_ptr<RemoteJobManager>
FacilityInfo::getRemoteJobManager(const std::string &name) const {
auto it = m_computeResources.find(name);
if (it == m_computeResources.end()) {
return boost::shared_ptr<RemoteJobManager>(); // TODO: should we throw an
// exception instead??
}
return (*it).second;
}
Roman Tolchenov
committed
} // namespace Kernel
} // namespace Mantid