meta_util.hpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. #ifndef REST_RPC_META_UTIL_HPP
  2. #define REST_RPC_META_UTIL_HPP
  3. #include "cplusplus_14.h"
  4. namespace rest_rpc{
  5. template <typename... Args, typename Func, std::size_t... Idx>
  6. void for_each(const std::tuple<Args...>& t, Func&& f, std::index_sequence<Idx...>) {
  7. (void)std::initializer_list<int> { (f(std::get<Idx>(t)), void(), 0)...};
  8. }
  9. template <typename... Args, typename Func, std::size_t... Idx>
  10. void for_each_i(const std::tuple<Args...>& t, Func&& f, std::index_sequence<Idx...>) {
  11. (void)std::initializer_list<int> { (f(std::get<Idx>(t), std::integral_constant<size_t, Idx>{}), void(), 0)...};
  12. }
  13. template<typename T>
  14. struct function_traits;
  15. template<typename Ret, typename Arg, typename... Args>
  16. struct function_traits<Ret(Arg, Args...)>
  17. {
  18. public:
  19. enum { arity = sizeof...(Args)+1 };
  20. typedef Ret function_type(Arg, Args...);
  21. typedef Ret return_type;
  22. using stl_function_type = std::function<function_type>;
  23. typedef Ret(*pointer)(Arg, Args...);
  24. typedef std::tuple<Arg, Args...> tuple_type;
  25. typedef std::tuple<std::remove_const_t<std::remove_reference_t<Arg>>, std::remove_const_t<std::remove_reference_t<Args>>...> bare_tuple_type;
  26. using args_tuple = std::tuple<std::string, Arg, std::remove_const_t<std::remove_reference_t<Args>>...>;
  27. using args_tuple_2nd = std::tuple<std::string, std::remove_const_t<std::remove_reference_t<Args>>...>;
  28. };
  29. template<typename Ret>
  30. struct function_traits<Ret()> {
  31. public:
  32. enum { arity = 0 };
  33. typedef Ret function_type();
  34. typedef Ret return_type;
  35. using stl_function_type = std::function<function_type>;
  36. typedef Ret(*pointer)();
  37. typedef std::tuple<> tuple_type;
  38. typedef std::tuple<> bare_tuple_type;
  39. using args_tuple = std::tuple<std::string>;
  40. using args_tuple_2nd = std::tuple<std::string>;
  41. };
  42. template<typename Ret, typename... Args>
  43. struct function_traits<Ret(*)(Args...)> : function_traits<Ret(Args...)>{};
  44. template <typename Ret, typename... Args>
  45. struct function_traits<std::function<Ret(Args...)>> : function_traits<Ret(Args...)>{};
  46. template <typename ReturnType, typename ClassType, typename... Args>
  47. struct function_traits<ReturnType(ClassType::*)(Args...)> : function_traits<ReturnType(Args...)>{};
  48. template <typename ReturnType, typename ClassType, typename... Args>
  49. struct function_traits<ReturnType(ClassType::*)(Args...) const> : function_traits<ReturnType(Args...)>{};
  50. template<typename Callable>
  51. struct function_traits : function_traits<decltype(&Callable::operator())>{};
  52. template<typename T>
  53. using remove_const_reference_t = std::remove_const_t<std::remove_reference_t<T>>;
  54. template<size_t... Is>
  55. auto make_tuple_from_sequence(std::index_sequence<Is...>)->decltype(std::make_tuple(Is...)) {
  56. std::make_tuple(Is...);
  57. }
  58. template<size_t N>
  59. constexpr auto make_tuple_from_sequence()->decltype(make_tuple_from_sequence(std::make_index_sequence<N>{})) {
  60. return make_tuple_from_sequence(std::make_index_sequence<N>{});
  61. }
  62. namespace detail {
  63. template <class Tuple, class F, std::size_t...Is>
  64. void tuple_switch(const std::size_t i, Tuple&& t, F&& f, std::index_sequence<Is...>) {
  65. (void)std::initializer_list<int> {
  66. (i == Is && (
  67. (void)std::forward<F>(f)(std::integral_constant<size_t, Is>{}), 0))...
  68. };
  69. }
  70. } // namespace detail
  71. template <class Tuple, class F>
  72. inline void tuple_switch(const std::size_t i, Tuple&& t, F&& f) {
  73. constexpr auto N =
  74. std::tuple_size<std::remove_reference_t<Tuple>>::value;
  75. detail::tuple_switch(i, std::forward<Tuple>(t), std::forward<F>(f),
  76. std::make_index_sequence<N>{});
  77. }
  78. template<int N, typename... Args>
  79. using nth_type_of = std::tuple_element_t<N, std::tuple<Args...>>;
  80. template<typename... Args>
  81. using last_type_of = nth_type_of<sizeof...(Args)-1, Args...>;
  82. }
  83. #endif //REST_RPC_META_UTIL_HPP