Skip to content
Snippets Groups Projects
Commit 889b9d44 authored by Felix Weiglhofer's avatar Felix Weiglhofer
Browse files

algo: Rework SerializableEnum.h

- Renamed header to EnumDict.h
- Implementation detail are now hidden with `CBM_ENUM_DICT`, this also
  simplifies enum declarations
- Use std::vector underneath instead of std::set
parent f1bd05aa
No related branches found
No related tags found
1 merge request!1175algo: Allow reading enums as strings from YAML.
......@@ -142,6 +142,8 @@ SpacesInContainerLiterals: false
SpacesInParentheses: false
SpacesInSquareBrackets: false
WhitespaceSensitiveMacros: ['CBM_ENUM_DICT']
Standard: c++11
TabWidth: 8
......
......@@ -8,7 +8,7 @@
#include "Prelude.h"
#include "config/Property.h"
#include "util/SerializableEnum.h"
#include "util/EnumDict.h"
namespace cbm::algo
{
......@@ -97,27 +97,16 @@ namespace cbm::algo
static constexpr auto Properties = std::make_tuple(config::Property(&RecoParams::sts, "sts", "STS reco settings"));
};
template<>
struct EnumIsSerializable<RecoParams::SortMode> : std::true_type {
};
template<>
inline const EnumDict_t<RecoParams::SortMode>& EnumDict<RecoParams::SortMode> = {
{"BlockSort", RecoParams::SortMode::BlockSort},
{"CUBSegmentedSort", RecoParams::SortMode::CUBSegmentedSort},
};
template<>
struct EnumIsSerializable<RecoParams::UnpackMode> : std::true_type {
};
template<>
inline const EnumDict_t<RecoParams::UnpackMode>& EnumDict<RecoParams::UnpackMode> = {
{"CPU", RecoParams::UnpackMode::CPU},
{"XPU", RecoParams::UnpackMode::XPU},
};
}; // namespace cbm::algo
CBM_ENUM_DICT(cbm::algo::RecoParams::SortMode,
{"BlockSort", RecoParams::SortMode::BlockSort},
{"CUBSegmentedSort", RecoParams::SortMode::CUBSegmentedSort}
);
}; // namespace cbm::algo
CBM_ENUM_DICT(cbm::algo::RecoParams::UnpackMode,
{"CPU", RecoParams::UnpackMode::CPU},
{"XPU", RecoParams::UnpackMode::XPU}
);
#endif // CBM_ALGO_BASE_RECOPARAMS_H
......@@ -13,7 +13,7 @@
#include "BaseTypes.h"
#include "Prelude.h"
#include "Property.h"
#include "util/SerializableEnum.h"
#include "util/EnumDict.h"
namespace cbm::algo::config
{
......@@ -29,8 +29,7 @@ namespace cbm::algo::config
{
using Type = std::remove_cv_t<std::remove_reference_t<T>>;
static_assert(!IsEnum<T> || EnumIsSerializable<Type>::value, "Enum must be serializable");
static_assert(!IsEnum<T> || detail::EnumHasDict_v<T>, "Enum must have a dictionary to be deserializable");
// TODO: error handling
if constexpr (IsBaseType<Type>) { return node.as<Type>(); }
......@@ -123,7 +122,7 @@ namespace cbm::algo::config
template<typename T>
void DoDump(const T& object, YAML::Emitter& ss, std::optional<YAML::EMITTER_MANIP> formatEntries = {})
{
static_assert(!IsEnum<T> || EnumIsSerializable<T>::value, "Enum must be serializable");
static_assert(!IsEnum<T> || detail::EnumHasDict_v<T>, "Enum must have a dictionary");
if constexpr (IsBaseType<T>) { ss << object; }
else if constexpr (IsEnum<T>) {
......
......@@ -9,21 +9,57 @@
#include <algorithm>
#include <iostream>
#include <optional>
#include <set>
#include <stdexcept>
#include <string_view>
#include <vector>
#include <fmt/format.h>
namespace cbm::algo
{
/**
* @brief Helper class to serialize enums to strings and back.
*
* @tparam T The enum type.
namespace detail
{
template<typename T>
using EnumDict_t = std::vector<std::pair<std::string_view, T>>;
template<typename T>
inline const EnumDict_t<T> EnumDict;
template<typename T>
struct EnumHasDict : std::false_type {
};
template<typename T>
inline constexpr bool EnumHasDict_v = EnumHasDict<T>::value;
} // namespace detail
template<typename T, typename = std::enable_if_t<detail::EnumHasDict_v<T>>>
std::optional<T> FromString(std::string_view str, bool caseSensitive = false)
{
const auto& dict = detail::EnumDict<T>;
auto it = std::find_if(dict.begin(), dict.end(), [&](const auto& pair) {
if (caseSensitive) return pair.first == str;
else
return boost::iequals(pair.first, str);
});
if (it == dict.end()) return std::nullopt;
return it->second;
}
template<typename T, typename = std::enable_if_t<detail::EnumHasDict_v<T>>>
std::string_view ToString(T t)
{
const auto& dict = detail::EnumDict<T>;
auto it = std::find_if(dict.begin(), dict.end(), [t](const auto& pair) { return pair.second == t; });
if (it == dict.end()) throw std::runtime_error(fmt::format("Entry {} for enum missing!", static_cast<int>(t)));
return it->first;
}
} // namespace cbm::algo
/**
* @brief Convert enums to strings and back.
*
* To use this class, you need to specialize EnumIsSerializable<T> and EnumStringMap<T>.
* Used to indicate that an enum can be converted to / from a string.
* @param type The enum type.
*
* Example:
* @code{.cpp}
......@@ -32,16 +68,10 @@ namespace cbm::algo
* TOF,
* };
*
* // Set to true to indicate that the enum is serializable.
* template<>
* struct EnumIsSerializable<Detector> : std::true_type {};
*
* // Create dictionary to convert enum to string and back.
* template<>
* const EnumDict_t<Detector> EnumDict<Detector> = {
* CBM_ENUM_DICT(Detector,
* {"sts", Detector::STS},
* {"tof", Detector::TOF},
* };
* {"tof", Detector::TOF}
* );
*
* // Use it like this:
* L_(info) << ToString(Detector::STS); // Prints "sts"
......@@ -50,51 +80,26 @@ namespace cbm::algo
* std::optional<Detector> d2 = FromString<Detector>("invalid"); // d2 == std::nullopt
* @endcode
*/
template<typename T>
struct EnumIsSerializable : std::false_type {
};
template<typename T>
using EnumDict_t = std::set<std::pair<std::string_view, T>>;
template<typename T>
inline const EnumDict_t<T> EnumDict;
template<typename T, typename = std::enable_if_t<EnumIsSerializable<T>::value>>
std::optional<T> FromString(std::string_view str, bool caseSensitive = false)
{
const auto& set = EnumDict<T>;
auto it = std::find_if(set.begin(), set.end(), [&](const auto& pair) {
if (caseSensitive) return pair.first == str;
else
return boost::iequals(pair.first, str);
});
if (it == set.end()) return std::nullopt;
return it->second;
#define CBM_ENUM_DICT(type, ...) \
template<> \
inline const cbm::algo::detail::EnumDict_t<type> cbm::algo::detail::EnumDict<type> = {__VA_ARGS__}; \
template<> \
struct cbm::algo::detail::EnumHasDict<type> : std::true_type { \
}
template<typename T, typename = std::enable_if_t<EnumIsSerializable<T>::value>>
std::string_view ToString(T t)
{
const auto& set = EnumDict<T>;
auto it = std::find_if(set.begin(), set.end(), [t](const auto& pair) { return pair.second == t; });
if (it == set.end()) throw std::runtime_error(fmt::format("Entry {} for enum missing!", static_cast<int>(t)));
return it->first;
}
} // namespace cbm::algo
// Stream operators for enums
// Placed in global namespace to be found by ADL e.g. for std::ostream_iterator
namespace std
{
template<typename T, typename = std::enable_if_t<cbm::algo::EnumIsSerializable<T>::value>>
template<typename T, typename = std::enable_if_t<cbm::algo::detail::EnumHasDict_v<T>>>
std::ostream& operator<<(std::ostream& os, T t)
{
os << cbm::algo::ToString(t);
return os;
}
template<typename T, typename = std::enable_if_t<cbm::algo::EnumIsSerializable<T>::value>>
template<typename T, typename = std::enable_if_t<cbm::algo::detail::EnumHasDict_v<T>>>
std::istream& operator>>(std::istream& is, T& t)
{
std::string str;
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment