mirror of
https://github.com/rdkit/rdkit.git
synced 2026-06-05 22:04:27 +08:00
231 lines
8.0 KiB
C++
231 lines
8.0 KiB
C++
// $Id$
|
|
//
|
|
// Copyright (C) 2003-2008 Greg Landrum and Rational Discovery LLC
|
|
//
|
|
// @@ All Rights Reserved @@
|
|
// This file is part of the RDKit.
|
|
// The contents are covered by the terms of the BSD license
|
|
// which is included in the file license.txt, found at the root
|
|
// of the RDKit source tree.
|
|
//
|
|
#include "Dict.h"
|
|
|
|
#include <boost/shared_array.hpp>
|
|
#include <boost/cstdint.hpp>
|
|
#include <boost/tuple/tuple.hpp>
|
|
#include <boost/utility.hpp>
|
|
#include <boost/type_traits.hpp>
|
|
#include <vector>
|
|
#include <list>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
#include <RDGeneral/LocaleSwitcher.h>
|
|
|
|
namespace RDKit {
|
|
namespace {
|
|
template <class T>
|
|
std::string vectToString(const boost::any &val) {
|
|
const std::vector<T> &tv = boost::any_cast<std::vector<T> >(val);
|
|
std::ostringstream sstr;
|
|
sstr << "[";
|
|
std::copy(tv.begin(), tv.end(), std::ostream_iterator<T>(sstr, ","));
|
|
sstr << "]";
|
|
return sstr.str();
|
|
}
|
|
std::string intVectToString(const boost::any &val) {
|
|
return vectToString<int>(val);
|
|
}
|
|
std::string uintVectToString(const boost::any &val) {
|
|
return vectToString<unsigned int>(val);
|
|
}
|
|
std::string doubleVectToString(const boost::any &val) {
|
|
// there is a performance cost associated with the
|
|
// LocaleSwitcher, so make sure we only instantiate it when
|
|
// it's really necessary (this was github #703)
|
|
Utils::LocaleSwitcher ls;
|
|
return vectToString<double>(val);
|
|
}
|
|
}
|
|
|
|
void Dict::getVal(const std::string &what, std::string &res) const {
|
|
//
|
|
// We're going to try and be somewhat crafty about this getVal stuff to make
|
|
// these
|
|
// containers a little bit more generic. The normal behavior here is that
|
|
// the
|
|
// value being queried must be directly castable to type T. We'll robustify
|
|
// a
|
|
// little bit by trying that and, if the cast fails, attempting a couple of
|
|
// other casts, which will then be lexically cast to type T.
|
|
//
|
|
if (!hasVal(what)) throw KeyErrorException(what);
|
|
const boost::any &val = _data.find(what)->second;
|
|
try {
|
|
res = boost::any_cast<std::string>(val);
|
|
} catch (const boost::bad_any_cast &) {
|
|
if (val.type() == typeid(int)) {
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<int>(val));
|
|
} else if (val.type() == typeid(unsigned int)) {
|
|
res =
|
|
boost::lexical_cast<std::string>(boost::any_cast<unsigned int>(val));
|
|
} else if (val.type() == typeid(long)) {
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<long>(val));
|
|
} else if (val.type() == typeid(unsigned long)) {
|
|
res =
|
|
boost::lexical_cast<std::string>(boost::any_cast<unsigned long>(val));
|
|
} else if (val.type() == typeid(float)) {
|
|
Utils::LocaleSwitcher ls; // late instantiation of LocaleSwitcher
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<float>(val));
|
|
} else if (val.type() == typeid(double)) {
|
|
Utils::LocaleSwitcher ls; // late instantiation of LocaleSwitcher
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<double>(val));
|
|
} else if (val.type() == typeid(const char *)) {
|
|
res = std::string(boost::any_cast<const char *>(val));
|
|
} else if (val.type() == typeid(std::vector<unsigned int>)) {
|
|
res = uintVectToString(val);
|
|
} else if (val.type() == typeid(std::vector<int>)) {
|
|
res = intVectToString(val);
|
|
} else {
|
|
throw;
|
|
}
|
|
}
|
|
};
|
|
|
|
bool Dict::getValIfPresent(const std::string &what, std::string &res) const {
|
|
//
|
|
// We're going to try and be somewhat crafty about this getVal stuff to make
|
|
// these
|
|
// containers a little bit more generic. The normal behavior here is that
|
|
// the
|
|
// value being queried must be directly castable to type T. We'll robustify
|
|
// a
|
|
// little bit by trying that and, if the cast fails, attempting a couple of
|
|
// other casts, which will then be lexically cast to type T.
|
|
//
|
|
DataType::const_iterator pos = _data.find(what);
|
|
if (pos == _data.end()) return false;
|
|
const boost::any &val = pos->second;
|
|
try {
|
|
res = boost::any_cast<std::string>(val);
|
|
} catch (const boost::bad_any_cast &) {
|
|
if (val.type() == typeid(int)) {
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<int>(val));
|
|
} else if (val.type() == typeid(unsigned int)) {
|
|
res =
|
|
boost::lexical_cast<std::string>(boost::any_cast<unsigned int>(val));
|
|
} else if (val.type() == typeid(long)) {
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<long>(val));
|
|
} else if (val.type() == typeid(unsigned long)) {
|
|
res =
|
|
boost::lexical_cast<std::string>(boost::any_cast<unsigned long>(val));
|
|
} else if (val.type() == typeid(float)) {
|
|
Utils::LocaleSwitcher ls; // late instantiation of LocaleSwitcher
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<float>(val));
|
|
} else if (val.type() == typeid(double)) {
|
|
Utils::LocaleSwitcher ls; // late instantiation of LocaleSwitcher
|
|
res = boost::lexical_cast<std::string>(boost::any_cast<double>(val));
|
|
} else if (val.type() == typeid(const char *)) {
|
|
res = std::string(boost::any_cast<const char *>(val));
|
|
} else if (val.type() == typeid(std::vector<unsigned int>)) {
|
|
res = uintVectToString(val);
|
|
} else if (val.type() == typeid(std::vector<int>)) {
|
|
res = intVectToString(val);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
namespace {
|
|
template <class T>
|
|
typename boost::enable_if<boost::is_floating_point<T>, T>::type _fromany(
|
|
const boost::any &arg) {
|
|
T res;
|
|
if (arg.type() == typeid(std::string) || arg.type() == typeid(const char *)) {
|
|
Utils::LocaleSwitcher ls; // late instantiation of LocaleSwitcher
|
|
try {
|
|
res = boost::any_cast<T>(arg);
|
|
} catch (const boost::bad_any_cast &exc) {
|
|
try {
|
|
res = boost::lexical_cast<T>(boost::any_cast<std::string>(arg));
|
|
} catch (...) {
|
|
throw exc;
|
|
}
|
|
}
|
|
} else {
|
|
res = boost::any_cast<T>(arg);
|
|
}
|
|
return res;
|
|
}
|
|
template <class T>
|
|
typename boost::enable_if<boost::is_integral<T>, T>::type _fromany(
|
|
const boost::any &arg) {
|
|
T res;
|
|
if (arg.type() == typeid(std::string) || arg.type() == typeid(const char *)) {
|
|
try {
|
|
res = boost::any_cast<T>(arg);
|
|
} catch (const boost::bad_any_cast &exc) {
|
|
try {
|
|
res = boost::lexical_cast<T>(boost::any_cast<std::string>(arg));
|
|
} catch (...) {
|
|
throw exc;
|
|
}
|
|
}
|
|
} else {
|
|
res = boost::any_cast<T>(arg);
|
|
}
|
|
return res;
|
|
}
|
|
template <class T>
|
|
typename boost::disable_if<boost::is_arithmetic<T>, T>::type _fromany(
|
|
const boost::any &arg) {
|
|
return boost::any_cast<T>(arg);
|
|
}
|
|
}
|
|
template <typename T>
|
|
T Dict::fromany(const boost::any &arg) const {
|
|
return _fromany<T>(arg);
|
|
};
|
|
template <typename T>
|
|
boost::any Dict::toany(T arg) const {
|
|
return boost::any(arg);
|
|
};
|
|
|
|
#define ANY_FORCE(T) \
|
|
template T Dict::fromany<T>(const boost::any &arg) const; \
|
|
template boost::any Dict::toany<T>(T arg) const;
|
|
|
|
ANY_FORCE(bool);
|
|
ANY_FORCE(boost::shared_array<double>);
|
|
ANY_FORCE(boost::shared_array<int>);
|
|
ANY_FORCE(double);
|
|
ANY_FORCE(int);
|
|
ANY_FORCE(std::list<int>);
|
|
ANY_FORCE(std::string);
|
|
ANY_FORCE(std::vector<boost::shared_array<double> >);
|
|
ANY_FORCE(std::vector<boost::shared_array<int> >);
|
|
ANY_FORCE(std::vector<double>);
|
|
ANY_FORCE(std::vector<int>);
|
|
ANY_FORCE(std::vector<std::list<int> >);
|
|
ANY_FORCE(std::vector<std::string>);
|
|
ANY_FORCE(std::vector<std::vector<double> >);
|
|
ANY_FORCE(std::vector<std::vector<int> >);
|
|
ANY_FORCE(std::vector<unsigned int>);
|
|
ANY_FORCE(std::vector<unsigned long long>);
|
|
ANY_FORCE(unsigned int);
|
|
|
|
template const std::string &Dict::fromany<const std::string &>(
|
|
const boost::any &arg) const;
|
|
|
|
typedef boost::tuples::tuple<boost::uint32_t, boost::uint32_t, boost::uint32_t>
|
|
uint32_t_tuple;
|
|
typedef boost::tuples::tuple<double, double, double> double_tuple;
|
|
|
|
template uint32_t_tuple Dict::fromany<uint32_t_tuple>(
|
|
const boost::any &arg) const;
|
|
template boost::any Dict::toany<uint32_t_tuple>(uint32_t_tuple arg) const;
|
|
template double_tuple Dict::fromany<double_tuple>(const boost::any &arg) const;
|
|
template boost::any Dict::toany<double_tuple>(double_tuple arg) const;
|
|
}
|