"docs/git@code.ornl.gov:mantidproject/mantid.git" did not exist on "e2323c559e737fa5ce48b8bec7217c68ce318adc"
Newer
Older
//----------------------------------------------------------------------
// Includes
//----------------------------------------------------------------------
#include "MantidKernel/ConfigService.h"
#include "MantidKernel/DateAndTime.h"
#include "MantidKernel/MantidVersion.h"
Janik Zikovsky
committed
#include "MantidKernel/Strings.h"
#include "MantidKernel/Logger.h"
#include "MantidKernel/FilterChannel.h"
#include "MantidKernel/StdoutChannel.h"
Roman Tolchenov
committed
#include "MantidKernel/FacilityInfo.h"
#include "MantidKernel/NetworkProxy.h"
#include <Poco/Util/LoggingConfigurator.h>
#include <Poco/Util/SystemConfiguration.h>
#include <Poco/Util/PropertyFileConfiguration.h>
#include <Poco/LoggingFactory.h>
#include <Poco/Path.h>
#include <Poco/File.h>
#include <Poco/StringTokenizer.h>
#include <Poco/DOM/DOMParser.h>
#include <Poco/DOM/Document.h>
#include <Poco/DOM/NodeList.h>
#include <Poco/Environment.h>
Robert Whitley
committed
#include <Poco/Process.h>
#include <Poco/Logger.h>
#include <Poco/Channel.h>
#include <Poco/SplitterChannel.h>
#include <Poco/LoggingRegistry.h>
#include <Poco/PipeStream.h>
#include <Poco/StreamCopier.h>
Gigg, Martyn Anthony
committed
#include <boost/algorithm/string/replace.hpp>
Gigg, Martyn Anthony
committed
#include <boost/algorithm/string/join.hpp>
Gigg, Martyn Anthony
committed
#include <fstream>
#include <iostream>
Janik Zikovsky
committed
/**
* Get the welcome message for Mantid.
* @returns A string containing the welcome message for Mantid.
*/
std::string welcomeMessage() {
return "Welcome to Mantid " +
std::string(Mantid::Kernel::MantidVersion::version()) +
"\nPlease cite: " + Mantid::Kernel::MantidVersion::paperCitation() +
" and this release: " + Mantid::Kernel::MantidVersion::doi();
Janik Zikovsky
committed
}
namespace { // anonymous namespace for some utility functions
/// static Logger object
Logger g_log("ConfigService");
/**
* Split the supplied string on semicolons.
*
* @param path The path to split.
* @param splitted vector to put the splitted path into.
*/
void splitPath(const std::string &path, std::vector<std::string> &splitted) {
if (path.find(";") == std::string::npos) { // don't bother tokenizing
splitted.push_back(path);
return;
}
int options =
Poco::StringTokenizer::TOK_TRIM + Poco::StringTokenizer::TOK_IGNORE_EMPTY;
splitted.clear();
Poco::StringTokenizer tokenizer(path, ";,", options);
auto iend = tokenizer.end();
splitted.reserve(tokenizer.count());
for (auto itr = tokenizer.begin(); itr != iend; ++itr) {
splitted.push_back(*itr);
}
}
}
} // end of anonymous namespace
Campbell, Stuart
committed
/** Inner templated class to wrap the poco library objects that have protected
* destructors and expose them as public.
*/
template <typename T> class ConfigServiceImpl::WrappedObject : public T {
Campbell, Stuart
committed
public:
/// The template type of class that is being wrapped
typedef T element_type;
/// Simple constructor
WrappedObject() : T() { m_pPtr = static_cast<T *>(this); }
Campbell, Stuart
committed
/** Constructor with a class to wrap
Janik Zikovsky
committed
* @param F :: The object to wrap
template <typename Field> explicit WrappedObject(Field &F) : T(F) {
Campbell, Stuart
committed
}
Campbell, Stuart
committed
/// Copy constructor
WrappedObject(const WrappedObject<T> &A) : T(A) {
m_pPtr = static_cast<T *>(this);
Campbell, Stuart
committed
}
Campbell, Stuart
committed
/// Virtual destructor
Campbell, Stuart
committed
/// Overloaded * operator returns the wrapped object pointer
const T &operator*() const { return *m_pPtr; }
Campbell, Stuart
committed
/// Overloaded * operator returns the wrapped object pointer
Campbell, Stuart
committed
/// Overloaded -> operator returns the wrapped object pointer
const T *operator->() const { return m_pPtr; }
Campbell, Stuart
committed
/// Overloaded -> operator returns the wrapped object pointer
Campbell, Stuart
committed
private:
/// Private pointer to the wrapped class
Campbell, Stuart
committed
};
// Back to the ConfigService class itself...
Campbell, Stuart
committed
//-------------------------------
// Private member functions
//-------------------------------
Roman Tolchenov
committed
Campbell, Stuart
committed
/// Private constructor for singleton class
ConfigServiceImpl::ConfigServiceImpl()
: m_pConf(NULL), m_pSysConfig(NULL), m_changed_keys(), m_ConfigPaths(),
m_AbsolutePaths(), m_strBaseDir(""), m_PropertyString(""),
m_properties_file_name("Mantid.properties"),
#ifdef MPI_BUILD
// Use a different user properties file for an mpi-enabled build to avoid
// confusion if both are used on the same file system
m_user_properties_file_name("Mantid-mpi.user.properties"),
#else
m_user_properties_file_name("Mantid.user.properties"),
#endif
m_DataSearchDirs(), m_UserSearchDirs(), m_InstrumentDirs(),
m_instr_prefixes(), m_proxyInfo(), m_isProxySet(false) {
// getting at system details
m_pSysConfig = new WrappedObject<Poco::Util::SystemConfiguration>;
Campbell, Stuart
committed
m_pConf = 0;
// Register the FilterChannel with the Poco logging factory
Poco::LoggingFactory::defaultFactory().registerChannelClass(
"FilterChannel",
new Poco::Instantiator<Poco::FilterChannel, Poco::Channel>);
// Register StdChannel with Poco
Poco::LoggingFactory::defaultFactory().registerChannelClass(
"StdoutChannel",
new Poco::Instantiator<Poco::StdoutChannel, Poco::Channel>);
Campbell, Stuart
committed
Gigg, Martyn Anthony
committed
// Define the directory to search for the Mantid.properties file.
Janik Zikovsky
committed
Poco::File f;
// First directory: the current working
m_strBaseDir = Poco::Path::current();
f = Poco::File(m_strBaseDir + m_properties_file_name);
if (!f.exists()) {
// Check the executable directory to see if it includes a mantid.properties
// file
Gigg, Martyn Anthony
committed
m_strBaseDir = getDirectoryOfExecutable();
Janik Zikovsky
committed
f = Poco::File(m_strBaseDir + m_properties_file_name);
Janik Zikovsky
committed
// Last, use the MANTIDPATH environment var
if (Poco::Environment::has("MANTIDPATH")) {
// Here we have to follow the convention of the rest of this code and
// add a trailing slash.
// Note: adding it to the MANTIDPATH itself will make other parts of the
// code crash.
Janik Zikovsky
committed
m_strBaseDir = Poco::Environment::get("MANTIDPATH") + "/";
}
// Fill the list of possible relative path keys that may require conversion to
// absolute paths
m_ConfigPaths.insert(
std::make_pair("mantidqt.python_interfaces_directory", true));
Campbell, Stuart
committed
m_ConfigPaths.insert(std::make_pair("plugins.directory", true));
m_ConfigPaths.insert(std::make_pair("pvplugins.directory", true));
Russell Taylor
committed
m_ConfigPaths.insert(std::make_pair("mantidqt.plugins.directory", true));
Campbell, Stuart
committed
m_ConfigPaths.insert(std::make_pair("instrumentDefinition.directory", true));
m_ConfigPaths.insert(
std::make_pair("instrumentDefinition.vtpDirectory", true));
m_ConfigPaths.insert(std::make_pair("groupingFiles.directory", true));
m_ConfigPaths.insert(std::make_pair("maskFiles.directory", true));
m_ConfigPaths.insert(std::make_pair("colormaps.directory", true));
m_ConfigPaths.insert(
std::make_pair("requiredpythonscript.directories", true));
Campbell, Stuart
committed
m_ConfigPaths.insert(std::make_pair("pythonscripts.directory", true));
m_ConfigPaths.insert(std::make_pair("pythonscripts.directories", true));
m_ConfigPaths.insert(std::make_pair("python.plugins.directories", true));
m_ConfigPaths.insert(std::make_pair("user.python.plugins.directories", true));
m_ConfigPaths.insert(std::make_pair("datasearch.directories", true));
Campbell, Stuart
committed
m_ConfigPaths.insert(std::make_pair("icatDownload.directory", true));
// attempt to load the default properties file that resides in the directory
// of the executable
Janik Zikovsky
committed
std::string propertiesFilesList;
Gigg, Martyn Anthony
committed
updateConfig(getPropertiesDir() + m_properties_file_name, false, false);
Janik Zikovsky
committed
propertiesFilesList = getPropertiesDir() + m_properties_file_name;
// Load the local (machine) properties file, if it exists
Poco::File localFile(getLocalFilename());
updateConfig(getLocalFilename(), true, false);
propertiesFilesList += ", " + getLocalFilename();
}
if (Poco::Environment::has("MANTIDPROPERTIES")) {
// and then append the user properties
Janik Zikovsky
committed
updateConfig(getUserFilename(), true, false);
propertiesFilesList += ", " + getUserFilename();
// and the extra one from the environment
Janik Zikovsky
committed
updateConfig(Poco::Environment::get("MANTIDPROPERTIES"), true, true);
propertiesFilesList += ", " + Poco::Environment::get("MANTIDPROPERTIES");
Janik Zikovsky
committed
// Just do the user properties
updateConfig(getUserFilename(), true, true);
propertiesFilesList += ", " + getUserFilename();
}
Campbell, Stuart
committed
updateFacilities();
g_log.debug() << "ConfigService created." << std::endl;
g_log.debug() << "Configured Mantid.properties directory of application as "
<< getPropertiesDir() << std::endl;
g_log.information() << "This is Mantid version " << MantidVersion::version()
<< " revision " << MantidVersion::revision() << std::endl;
g_log.information() << "running on " << getComputerName() << " starting "
<< DateAndTime::getCurrentTime().toFormattedString(
"%Y-%m-%dT%H:%MZ") << "\n";
g_log.information() << "Properties file(s) loaded: " << propertiesFilesList
<< std::endl;
#ifndef MPI_BUILD // There is no logging to file by default in MPI build
g_log.information() << "Logging to: " << m_logFilePath << std::endl;
// Assert that the appdata and the instrument subdirectory exists
std::string appDataDir = getAppDataDir();
Poco::Path path(appDataDir);
path.pushDirectory("instrument");
Poco::File file(path);
// createDirectories will fail gracefully if it is already present - but will
// throw an error if it cannot create the directory
try {
file.createDirectories();
} catch (Poco::FileException &fe) {
g_log.error()
<< "Cannot create the local instrument cache directory ["
<< path.toString()
<< "]. Mantid will not be able to update instrument definitions.\n"
<< fe.what() << std::endl;
}
try {
vtpDir.createDirectories();
} catch (Poco::FileException &fe) {
g_log.error()
<< "Cannot create the local instrument geometry cache directory ["
<< path.toString()
<< "]. Mantid will be slower at viewing complex instruments.\n"
<< fe.what() << std::endl;
}
// must update the cache of instrument paths
cacheInstrumentPaths();
Campbell, Stuart
committed
}
/** Private Destructor
* Prevents client from calling 'delete' on the pointer handed out by Instance
*/
ConfigServiceImpl::~ConfigServiceImpl() {
// std::cerr << "ConfigService destroyed." << std::endl;
Campbell, Stuart
committed
Kernel::Logger::shutdown();
delete m_pSysConfig;
delete m_pConf; // potential double delete???
clearFacilities();
Campbell, Stuart
committed
}
/** Loads the config file provided.
* If the file contains logging setup instructions then these will be used to
*setup the logging framework.
Campbell, Stuart
committed
*
Janik Zikovsky
committed
* @param filename :: The filename and optionally path of the file to load
* @param append :: If false (default) then any previous configuration is
*discarded, otherwise the new keys are added, and repeated keys will override
*existing ones.
Campbell, Stuart
committed
*/
void ConfigServiceImpl::loadConfig(const std::string &filename,
const bool append) {
Campbell, Stuart
committed
delete m_pConf;
if (!append) {
// remove the previous property string
Campbell, Stuart
committed
m_PropertyString = "";
m_changed_keys.clear();
Campbell, Stuart
committed
std::string temp;
bool good = readFile(filename, temp);
Campbell, Stuart
committed
// check if we have failed to open the file
if ((!good) || (temp == "")) {
if (filename == getUserPropertiesDir() + m_user_properties_file_name) {
// write out a fresh file
Campbell, Stuart
committed
createUserPropertiesFile();
Campbell, Stuart
committed
throw Exception::FileError("Cannot open file", filename);
Campbell, Stuart
committed
}
// store the property string
if ((append) && (m_PropertyString != "")) {
Campbell, Stuart
committed
m_PropertyString = m_PropertyString + "\n" + temp;
Campbell, Stuart
committed
m_PropertyString = temp;
} catch (std::exception &e) {
// there was a problem loading the file - it probably is not there
std::cerr << "Problem loading the configuration file " << filename << " "
<< e.what() << std::endl;
if (!append) {
Campbell, Stuart
committed
// if we have no property values then take the default
m_PropertyString = defaultConfig();
Campbell, Stuart
committed
}
// use the cached property string to initialise the POCO property file
Campbell, Stuart
committed
std::istringstream istr(m_PropertyString);
m_pConf = new WrappedObject<Poco::Util::PropertyFileConfiguration>(istr);
Campbell, Stuart
committed
}
/**
* Read a file and place its contents into the given string
Janik Zikovsky
committed
* @param filename :: The filename of the file to read
* @param contents :: The file contents will be placed here
Campbell, Stuart
committed
* @returns A boolean indicating whether opening the file was successful
*/
bool ConfigServiceImpl::readFile(const std::string &filename,
std::string &contents) const {
Campbell, Stuart
committed
std::ifstream propFile(filename.c_str(), std::ios::in);
bool good = propFile.good();
Loading
Loading full blame...