writable_field_traits.hpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. //
  2. // Copyright (c) 2019-2023 Ruben Perez Hidalgo (rubenperez038 at gmail dot com)
  3. //
  4. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  5. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  6. //
  7. #ifndef BHO_MYSQL_DETAIL_WRITABLE_FIELD_TRAITS_HPP
  8. #define BHO_MYSQL_DETAIL_WRITABLE_FIELD_TRAITS_HPP
  9. #include <asio2/bho/mysql/field_view.hpp>
  10. #include <asio2/bho/mysql/detail/config.hpp>
  11. #include <type_traits>
  12. namespace bho {
  13. namespace mysql {
  14. namespace detail {
  15. template <class T, class En1 = void, class En2 = void>
  16. struct writable_field_traits
  17. {
  18. static constexpr bool is_supported = false;
  19. };
  20. template <>
  21. struct writable_field_traits<bool, void, void>
  22. {
  23. static constexpr bool is_supported = true;
  24. static field_view to_field(bool value) noexcept { return field_view(value ? 1 : 0); }
  25. };
  26. template <class T>
  27. struct writable_field_traits<
  28. T,
  29. typename std::enable_if<std::is_constructible<field_view, const T&>::value>::type,
  30. void>
  31. {
  32. static constexpr bool is_supported = true;
  33. static field_view to_field(const T& value) noexcept { return field_view(value); }
  34. };
  35. // Optionals. To avoid dependencies, we use a "concept".
  36. // We consider a type an optional if has a `bool has_value() const` and
  37. // `const value_type& value() const`
  38. template <class T>
  39. struct writable_field_traits<
  40. T,
  41. void,
  42. typename std::enable_if<
  43. std::is_same<decltype(std::declval<const T&>().has_value()), bool>::value &&
  44. std::is_same<decltype(std::declval<const T&>().value()), const typename T::value_type&>::value>::type>
  45. {
  46. using value_traits = writable_field_traits<typename T::value_type>;
  47. static constexpr bool is_supported = value_traits::is_supported;
  48. static field_view to_field(const T& value) noexcept
  49. {
  50. return value.has_value() ? value_traits::to_field(value.value()) : field_view();
  51. }
  52. };
  53. template <class T>
  54. field_view to_field(const T& value) noexcept
  55. {
  56. return writable_field_traits<T>::to_field(value);
  57. }
  58. template <class T>
  59. struct is_writable_field
  60. {
  61. static constexpr bool value = writable_field_traits<T>::is_supported;
  62. };
  63. // field_view_forward_iterator
  64. template <typename T, typename = void>
  65. struct is_field_view_forward_iterator : std::false_type
  66. {
  67. };
  68. // clang-format off
  69. template <typename T>
  70. struct is_field_view_forward_iterator<
  71. T,
  72. typename std::enable_if<
  73. std::is_convertible<
  74. typename std::iterator_traits<T>::reference,
  75. field_view
  76. >::value
  77. &&
  78. std::is_base_of<
  79. std::forward_iterator_tag,
  80. typename std::iterator_traits<T>::iterator_category
  81. >::value
  82. >::type
  83. > : std::true_type { };
  84. // clang-format on
  85. #ifdef BHO_MYSQL_HAS_CONCEPTS
  86. template <class T>
  87. concept field_view_forward_iterator = is_field_view_forward_iterator<T>::value;
  88. #define BHO_MYSQL_FIELD_VIEW_FORWARD_ITERATOR ::bho::mysql::detail::field_view_forward_iterator
  89. #else // BHO_MYSQL_HAS_CONCEPTS
  90. #define BHO_MYSQL_FIELD_VIEW_FORWARD_ITERATOR class
  91. #endif // BHO_MYSQL_HAS_CONCEPTS
  92. // writable_field_tuple
  93. template <class... T>
  94. struct is_writable_field_tuple_impl : std::false_type
  95. {
  96. };
  97. template <class... T>
  98. struct is_writable_field_tuple_impl<std::tuple<T...>>
  99. : mp11::mp_all_of<mp11::mp_list<T...>, is_writable_field>
  100. {
  101. };
  102. template <class Tuple>
  103. struct is_writable_field_tuple : is_writable_field_tuple_impl<typename std::decay<Tuple>::type>
  104. {
  105. };
  106. #ifdef BHO_MYSQL_HAS_CONCEPTS
  107. template <class T>
  108. concept writable_field_tuple = is_writable_field_tuple<T>::value;
  109. #define BHO_MYSQL_WRITABLE_FIELD_TUPLE ::bho::mysql::detail::writable_field_tuple
  110. #else // BHO_MYSQL_HAS_CONCEPTS
  111. #define BHO_MYSQL_WRITABLE_FIELD_TUPLE class
  112. #endif // BHO_MYSQL_HAS_CONCEPTS
  113. } // namespace detail
  114. } // namespace mysql
  115. } // namespace bho
  116. #endif