-
-
Notifications
You must be signed in to change notification settings - Fork 185
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Recursive traversal of types to generate required pools
- Loading branch information
1 parent
fab1e15
commit 023531f
Showing
5 changed files
with
227 additions
and
84 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,162 @@ | ||
#pragma once | ||
|
||
#include <immer/flex_vector.hpp> | ||
#include <immer/map.hpp> | ||
#include <immer/set.hpp> | ||
#include <immer/table.hpp> | ||
#include <immer/vector.hpp> | ||
|
||
#include <boost/hana.hpp> | ||
|
||
namespace immer::persist::util { | ||
|
||
namespace detail { | ||
|
||
namespace hana = boost::hana; | ||
|
||
template <class T, class = void> | ||
struct get_inner_types_t | ||
{ | ||
static auto apply() | ||
{ | ||
return hana::make_tuple( | ||
hana::make_pair(hana::type_c<T>, BOOST_HANA_STRING(""))); | ||
} | ||
}; | ||
|
||
template <class T> | ||
struct get_inner_types_t<T, std::enable_if_t<hana::Struct<T>::value>> | ||
{ | ||
static auto apply() | ||
{ | ||
auto value = T{}; | ||
return hana::transform(hana::keys(value), [&](auto key) { | ||
const auto& member = hana::at_key(value, key); | ||
return hana::make_pair(hana::typeid_(member), key); | ||
}); | ||
} | ||
}; | ||
|
||
template <typename T, | ||
typename MemoryPolicy, | ||
immer::detail::rbts::bits_t B, | ||
immer::detail::rbts::bits_t BL> | ||
struct get_inner_types_t<immer::vector<T, MemoryPolicy, B, BL>> | ||
: get_inner_types_t<T> | ||
{}; | ||
|
||
template <typename T, | ||
typename MemoryPolicy, | ||
immer::detail::rbts::bits_t B, | ||
immer::detail::rbts::bits_t BL> | ||
struct get_inner_types_t<immer::flex_vector<T, MemoryPolicy, B, BL>> | ||
: get_inner_types_t<T> | ||
{}; | ||
|
||
template <typename T, typename MemoryPolicy> | ||
struct get_inner_types_t<immer::box<T, MemoryPolicy>> : get_inner_types_t<T> | ||
{}; | ||
|
||
template <typename T, | ||
typename Hash, | ||
typename Equal, | ||
typename MemoryPolicy, | ||
immer::detail::hamts::bits_t B> | ||
struct get_inner_types_t<immer::set<T, Hash, Equal, MemoryPolicy, B>> | ||
: get_inner_types_t<T> | ||
{}; | ||
|
||
template <typename K, | ||
typename T, | ||
typename Hash, | ||
typename Equal, | ||
typename MemoryPolicy, | ||
immer::detail::hamts::bits_t B> | ||
struct get_inner_types_t<immer::map<K, T, Hash, Equal, MemoryPolicy, B>> | ||
{ | ||
static auto apply() | ||
{ | ||
return hana::concat(get_inner_types_t<K>::apply(), | ||
get_inner_types_t<T>::apply()); | ||
} | ||
}; | ||
|
||
template <typename T, | ||
typename KeyFn, | ||
typename Hash, | ||
typename Equal, | ||
typename MemoryPolicy, | ||
immer::detail::hamts::bits_t B> | ||
struct get_inner_types_t<immer::table<T, KeyFn, Hash, Equal, MemoryPolicy, B>> | ||
: get_inner_types_t<T> | ||
{}; | ||
|
||
template <class... Types> | ||
struct get_inner_types_t<std::variant<Types...>> | ||
{ | ||
static auto apply() | ||
{ | ||
return hana::make_tuple( | ||
hana::make_pair(hana::type_c<Types>, BOOST_HANA_STRING(""))...); | ||
} | ||
}; | ||
|
||
constexpr auto insert_conditionally = [](auto map, auto pair) { | ||
namespace hana = hana; | ||
using contains_t = | ||
decltype(hana::is_just(hana::find(map, hana::first(pair)))); | ||
if constexpr (contains_t::value) { | ||
const auto empty_string = BOOST_HANA_STRING(""); | ||
using is_empty_t = decltype(hana::second(pair) == empty_string); | ||
if constexpr (is_empty_t::value) { | ||
// Do not replace with empty string | ||
return map; | ||
} else { | ||
return hana::insert(map, pair); | ||
} | ||
} else { | ||
return hana::insert(map, pair); | ||
} | ||
}; | ||
|
||
} // namespace detail | ||
|
||
/** | ||
* Generate a map (type, member_name) for all members of a given type, | ||
* recursively. | ||
*/ | ||
inline auto get_inner_types(const auto& type) | ||
{ | ||
namespace hana = boost::hana; | ||
|
||
constexpr auto get_for_one_type = [](auto type) { | ||
using T = typename decltype(type)::type; | ||
return detail::get_inner_types_t<T>::apply(); | ||
}; | ||
|
||
constexpr auto get_for_many = [get_for_one_type](const auto& map) { | ||
auto new_pairs = hana::to_tuple(map) | [get_for_one_type](auto pair) { | ||
return get_for_one_type(hana::first(pair)); | ||
}; | ||
|
||
return hana::fold_left( | ||
hana::to_map(new_pairs), map, detail::insert_conditionally); | ||
}; | ||
|
||
constexpr auto can_expand = [get_for_many](const auto& set) { | ||
return set != get_for_many(set); | ||
}; | ||
|
||
auto expanded = hana::while_( | ||
can_expand, hana::to_map(get_for_one_type(type)), get_for_many); | ||
|
||
// Throw away types we don't know names for | ||
const auto empty_string = BOOST_HANA_STRING(""); | ||
auto result = | ||
hana::filter(hana::to_tuple(expanded), [empty_string](auto pair) { | ||
return hana::second(pair) != empty_string; | ||
}); | ||
return hana::to_map(result); | ||
} | ||
|
||
} // namespace immer::persist::util |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.