diff --git a/algo/base/config/BaseTypes.h b/algo/base/config/BaseTypes.h index 5cb598dd7f5b77df65a48c10de4d9966a487996a..1f2701f67c71b8eb14e40720126a0f4654ee5aec 100644 --- a/algo/base/config/BaseTypes.h +++ b/algo/base/config/BaseTypes.h @@ -13,7 +13,7 @@ namespace cbm::algo::config { - using BaseTypes = std::tuple<bool, u8, i8, u16, i16, u32, i32, u64, i64, f32, f64, std::string>; + using FundamentalTypes = std::tuple<bool, u8, i8, u16, i16, u32, i32, u64, i64, f32, f64, std::string>; template<typename T, typename Tuple> struct has_type; @@ -23,13 +23,16 @@ namespace cbm::algo::config }; template<typename T> - constexpr bool IsBaseType = has_type<T, BaseTypes>::value; + constexpr bool IsFundamental = has_type<T, FundamentalTypes>::value; template<typename T> constexpr bool IsEnum = std::is_enum_v<T>; - template<typename T, typename = std::enable_if_t<IsBaseType<T>>> - constexpr std::string_view BaseTypeToStr() + template<typename T> + constexpr bool IsScalar = IsFundamental<T> || IsEnum<T>; + + template<typename T, typename = std::enable_if_t<IsFundamental<T>>> + constexpr std::string_view Typename() { if constexpr (std::is_same_v<bool, T>) { return "bool"; } else if constexpr (std::is_same_v<u8, T>) { @@ -50,8 +53,11 @@ namespace cbm::algo::config else if constexpr (std::is_same_v<i32, T>) { return "i32"; } - else if constexpr (std::is_same_v<float, T>) { - return "float"; + else if constexpr (std::is_same_v<f32, T>) { + return "f32"; + } + else if constexpr (std::is_same_v<f64, T>) { + return "f64"; } else if constexpr (std::is_same_v<std::string, T>) { return "string"; @@ -83,6 +89,17 @@ namespace cbm::algo::config template<typename T> constexpr bool IsArray = is_std_array<T>::value; + template<typename> + struct is_std_map : std::false_type { + }; + + template<typename K, typename V, typename C, typename A> + struct is_std_map<std::map<K, V, C, A>> : std::true_type { + }; + + template<typename T> + constexpr bool IsMap = is_std_map<T>::value; + } // namespace cbm::algo::config #endif // CBM_ALGO_BASE_CONFIG_BASETYPES_H diff --git a/algo/base/config/Property.h b/algo/base/config/Property.h index b099498b118e5fb7042eaea1dca2626c2121d7ea..f0ef1ffadb1ee6d404c357d14259c7ef03ffb4a2 100644 --- a/algo/base/config/Property.h +++ b/algo/base/config/Property.h @@ -63,4 +63,15 @@ namespace cbm::algo::config } // namespace cbm::algo::config +#define CBM_PROPERTIES(...) \ +public: \ + static constexpr auto Properties = std::make_tuple(__VA_ARGS__) + +/** + * @brief Optional tag to specify a formatting of the class (YAML::Flow vs YAML::Block) +*/ +#define CBM_FORMAT(tag) \ +public: \ + static constexpr std::optional<YAML::EMITTER_MANIP> FormatAs = tag + #endif // CBM_ALGO_BASE_CONFIG_PROPERTY_H diff --git a/algo/base/config/Yaml.h b/algo/base/config/Yaml.h index 71895d38ce8b208bc997c99d632eab084f38467d..50e6684eb05b54ad3be7f2a48708a88f05747fc6 100644 --- a/algo/base/config/Yaml.h +++ b/algo/base/config/Yaml.h @@ -24,6 +24,16 @@ namespace cbm::algo::config (func(std::integral_constant<T, Values> {}), ...); } + template<typename T, typename = void> + struct GetFmtTag { + static constexpr std::optional<YAML::EMITTER_MANIP> value = {}; + }; + + template<typename T> + struct GetFmtTag<T, std::void_t<decltype(T::FormatAs)>> { + static constexpr std::optional<YAML::EMITTER_MANIP> value = T::FormatAs; + }; + template<typename T> T Read(const YAML::Node& node) { @@ -32,7 +42,7 @@ namespace cbm::algo::config 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>(); } + if constexpr (IsFundamental<Type>) { return node.as<Type>(); } else if constexpr (IsEnum<Type>) { std::optional<T> maybet = FromString<T>(node.as<std::string>()); if (!maybet) { throw std::runtime_error(fmt::format("Invalid enum value: {}", node.as<std::string>())); } @@ -46,7 +56,7 @@ namespace cbm::algo::config return vector; } else if constexpr (IsArray<Type>) { - Type array {}; + Type array = {}; auto vector = Read<std::vector<typename Type::value_type>>(node); if (vector.size() != array.size()) { throw std::runtime_error(fmt::format("Array size mismatch: expected {}, got {}", array.size(), vector.size())); @@ -54,6 +64,20 @@ namespace cbm::algo::config std::copy(vector.begin(), vector.end(), array.begin()); return array; } + else if constexpr (IsMap<Type>) { + using Key_t = typename Type::key_type; + using Val_t = typename Type::mapped_type; + + static_assert(IsScalar<Key_t>, "Map key must be a fundamental or enum type"); + + Type map; + for (YAML::const_iterator it = node.begin(); it != node.end(); ++it) { + const auto& key = it->first; + const auto& value = it->second; + map[Read<Key_t>(key)] = Read<Val_t>(value); + } + return map; + } else { Type object; constexpr auto nProperties = std::tuple_size<decltype(Type::Properties)>::value; @@ -74,11 +98,11 @@ namespace cbm::algo::config std::stringstream docString; - if constexpr (IsBaseType<Type>) { docString << BaseTypeToStr<Type>(); } + if constexpr (IsFundamental<Type>) { docString << Typename<Type>(); } else if constexpr (IsVector<Type> || IsArray<Type>) { using ChildType = typename Type::value_type; - if constexpr (IsBaseType<ChildType>) { - docString << std::string(indent, ' ') << "list of " << BaseTypeToStr<ChildType>() << std::endl; + if constexpr (IsFundamental<ChildType>) { + docString << std::string(indent, ' ') << "list of " << Typename<ChildType>() << std::endl; } else { docString << std::string(indent, ' ') << "list of" << std::endl; @@ -91,9 +115,9 @@ namespace cbm::algo::config using ChildType = std::remove_cv_t< std::remove_reference_t<decltype(std::get<index>(Type::Properties).Get(std::declval<Type>()))>>; auto& property = std::get<index>(Type::Properties); - if constexpr (IsBaseType<ChildType>) { + if constexpr (IsFundamental<ChildType>) { docString << std::string(indent, ' ') << property.Key() << ": " << property.Description() << " [" - << BaseTypeToStr<ChildType>() << "]" << std::endl; + << Typename<ChildType>() << "]" << std::endl; } else { docString << std::string(indent, ' ') << property.Key() << ": " << property.Description() << std::endl; @@ -124,7 +148,12 @@ namespace cbm::algo::config { static_assert(!IsEnum<T> || detail::EnumHasDict_v<T>, "Enum must have a dictionary"); - if constexpr (IsBaseType<T>) { ss << object; } + if constexpr (IsFundamental<T>) { + // Take care that i8 and u8 are printed as integers not as characters + if constexpr (std::is_same_v<T, i8> || std::is_same_v<T, u8>) ss << i32(object); + else + ss << object; + } else if constexpr (IsEnum<T>) { ss << ToString<T>(object); } @@ -136,8 +165,19 @@ namespace cbm::algo::config } ss << YAML::EndSeq; } + else if constexpr (IsMap<T>) { + ss << YAML::BeginMap; + for (const auto& [key, value] : object) { + if (formatEntries.has_value()) { ss << formatEntries.value(); } + ss << YAML::Key << key; + ss << YAML::Value; + DoDump(value, ss); + } + ss << YAML::EndMap; + } else { constexpr auto nProperties = std::tuple_size<decltype(T::Properties)>::value; + if (auto fmtTag = GetFmtTag<T>::value; fmtTag.has_value()) { ss << fmtTag.value(); } ss << YAML::BeginMap; ForEach(std::make_integer_sequence<std::size_t, nProperties> {}, [&](auto index) { auto& property = std::get<index>(T::Properties); diff --git a/algo/test/CMakeLists.txt b/algo/test/CMakeLists.txt index 4cd6184ace6a94c6e681a92ea9d111a7048501c0..11caa166166b9badd1cd17d890cc0f550b97fff9 100644 --- a/algo/test/CMakeLists.txt +++ b/algo/test/CMakeLists.txt @@ -45,3 +45,9 @@ Set(DigiEventSelectorSources CreateGTestExeAndAddTest(_GTestDigiEventSelector "${INCLUDE_DIRECTORIES}" "${LINK_DIRECTORIES}" "${DigiEventSelectorSources}" "${PUB_DEPS}" "${PVT_DEPS}" "${INT_DEPS}" "") + +Set(YamlConfigSources + _GTestYamlConfig.cxx +) +CreateGTestExeAndAddTest(_GTestYamlConfig "${INCLUDE_DIRECTORIES}" "${LINK_DIRECTORIES}" + "${YamlConfigSources}" "${PUB_DEPS}" "${PVT_DEPS}" "${INT_DEPS}" "") diff --git a/algo/test/_GTestYamlConfig.cxx b/algo/test/_GTestYamlConfig.cxx new file mode 100644 index 0000000000000000000000000000000000000000..c0c35d327468720ea7fb825eed9fb7abc7677b27 --- /dev/null +++ b/algo/test/_GTestYamlConfig.cxx @@ -0,0 +1,73 @@ +/* Copyright (C) 2023 FIAS Frankfurt Institute for Advanced Studies, Frankfurt / Main + SPDX-License-Identifier: GPL-3.0-only + Authors: Felix Weiglhofer [committer] */ + +#include "config/Property.h" +#include "config/Yaml.h" +#include "gtest/gtest.h" + +using namespace cbm::algo; + +// +// TODO: These tests are nowhere near exhaustive. Extend after the DC. +// + +TEST(Config, CanDeserializeStdMap) +{ + using Map_t = std::map<i32, i32>; + + std::stringstream ss; + ss << "1: 2\n"; + ss << "3: 4\n"; + ss << "5: 6\n"; + + YAML::Node node = YAML::Load(ss.str()); + auto map = config::Read<Map_t>(node); + + EXPECT_EQ(map.size(), 3); + EXPECT_EQ(map.at(1), 2); + EXPECT_EQ(map.at(3), 4); + EXPECT_EQ(map.at(5), 6); +} + +TEST(Config, CanSerializeStdMap) +{ + using Map_t = std::map<i32, i32>; + + Map_t map; + map[1] = 2; + map[3] = 4; + map[5] = 6; + + YAML::Node node = YAML::Load(config::Dump {}(map)); + + EXPECT_EQ(node.size(), 3); + EXPECT_EQ(node[1].as<i32>(), 2); + EXPECT_EQ(node[3].as<i32>(), 4); + EXPECT_EQ(node[5].as<i32>(), 6); +} + +class Foo { +private: + i32 fBar = 42; + i32 fBaz = 43; + +public: + i32 GetBar() const { return fBar; } + i32 GetBaz() const { return fBaz; } + + CBM_PROPERTIES(config::Property(&Foo::fBar, "bar", ""), config::Property(&Foo::fBaz, "baz", "")); +}; + +TEST(Config, CanAccessPrivateMembers) +{ + std::stringstream ss; + ss << "bar: 1\n"; + ss << "baz: 2\n"; + + YAML::Node node = YAML::Load(ss.str()); + auto foo = config::Read<Foo>(node); + + EXPECT_EQ(foo.GetBar(), 1); + EXPECT_EQ(foo.GetBaz(), 2); +}