arg_router  1.4.0
C++ command line argument parsing and routing
detail.hpp
1 // Copyright (C) 2022-2023 by Camden Mannett.
2 // Distributed under the Boost Software License, Version 1.0.
3 // (See accompanying file LICENSE or copy at https://www.boost.org/LICENSE_1_0.txt)
4 
5 #pragma once
6 
7 #include "arg_router/policy/display_name.hpp"
8 #include "arg_router/policy/error_name.hpp"
9 #include "arg_router/policy/no_result_value.hpp"
10 #include "arg_router/tree_node.hpp"
11 
12 #include <variant>
13 
14 namespace arg_router::dependency::detail
15 {
16 // Collect the child names and concetenate into a compile time string for the display name
17 template <typename ParentDocName, typename... Params>
18 struct generate_string_of_child_names {
19 private:
20  template <typename Prefix, typename Strings>
21  using prepend_prefix =
22 #ifdef AR_ENABLE_CPP20_STRINGS
23  std::decay_t<decltype(
24  Prefix{} + boost::mp11::mp_fold<Strings, utility::str<>, utility::str_concat>{})>;
25 #else
26  boost::mp11::mp_flatten<boost::mp11::mp_insert_c<Strings, 0, typename Prefix::array_type>>;
27 #endif
28 
29  template <typename Child>
30  struct build_name {
31  // This is clunky, but it's still more readable than doing it with bare types
32  constexpr static auto build()
33  {
34  constexpr auto token = parsing::node_token_type<Child>();
35  return AR_STRING_SV(parsing::to_string(token.prefix)){} + AR_STRING_SV(token.name){};
36  }
37 
38  using type = typename std::decay_t<decltype(build())>;
39  };
40 
41  template <typename ChildName>
42  using joiner =
43 #ifdef AR_ENABLE_CPP20_STRINGS
44  typename ChildName::template append_t<utility::str<','>>;
45 #else
46  boost::mp11::mp_push_back<typename ChildName::array_type, traits::integral_constant<','>>;
47 #endif
48 
49  using children_type = boost::mp11::mp_filter<is_tree_node, std::tuple<Params...>>;
50 
51  // The gist is that for each child we get its name (prepended with an appropriate prefix) and
52  // concatenate with a comma separator. And then add a helpful prefix to the whole thing
53  using concatenated_string = prepend_prefix<
54  ParentDocName,
55  boost::mp11::mp_transform<
56  joiner,
57  boost::mp11::mp_transform_q<
58  boost::mp11::mp_bind<traits::get_type,
59  boost::mp11::mp_bind<build_name, boost::mp11::_1>>,
60  children_type>>>;
61 
62 public:
63 // We only take the first N-1 characters to strip off the trailing comma
64 #ifdef AR_ENABLE_CPP20_STRINGS
65  using type = std::decay_t<decltype(
66  concatenated_string{}.template substr<0, concatenated_string::size() - 1>())>;
67 #else
69  boost::mp11::mp_take_c<concatenated_string, std::tuple_size_v<concatenated_string> - 1>>;
70 #endif
71 };
72 
73 // If a display_name has been specified, then use that otherwise use the dev-provided default.
74 // Always generate an error string referring to the children
75 template <typename DefaultString, typename... Policies>
76 class add_names
77 {
78  template <typename Policy, typename Enable = void>
79  struct has_display_name_t : std::false_type {
80  };
81 
82  template <typename Policy>
83  struct has_display_name_t<Policy, std::enable_if_t<policy::is_policy_v<Policy>>> {
84  constexpr static bool value = traits::has_display_name_method_v<Policy>;
85  };
86 
87  using policies_tuple = std::tuple<std::decay_t<Policies>...>;
88 
89  using display_index = boost::mp11::mp_find_if<policies_tuple, has_display_name_t>;
90 
91  template <typename I>
92  using display_type = typename std::tuple_element_t<I::value, policies_tuple>::string_type;
93 
94 public:
95  constexpr static auto has_display_name =
96  display_index::value != std::tuple_size_v<policies_tuple>;
97 
98 private:
99  using display_string =
100  boost::mp11::mp_eval_if_c<!has_display_name, DefaultString, display_type, display_index>;
101 
102  using policies_tuple_with_error_name =
103  boost::mp11::mp_push_front<policies_tuple,
104  policy::error_name_t<typename generate_string_of_child_names<
105  display_string,
106  std::decay_t<Policies>...>::type>>;
107 
108 public:
109  using type = std::conditional_t<
110  has_display_name,
111  boost::mp11::mp_rename<policies_tuple_with_error_name, tree_node>,
112  boost::mp11::mp_rename<boost::mp11::mp_push_front<policies_tuple_with_error_name,
113  policy::display_name_t<DefaultString>>,
114  tree_node>>;
115 };
116 
117 template <typename ParentDocName, typename... Params>
118 class basic_one_of_t : public add_names<ParentDocName, Params...>::type
119 {
120  using parent_type = typename add_names<ParentDocName, Params...>::type;
121 
122  static_assert((std::tuple_size_v<typename parent_type::children_type> >= 2),
123  "basic_one_of_t must have at least one two child nodes");
124  static_assert(!traits::has_long_name_method_v<basic_one_of_t>,
125  "basic_one_of_t must not have a long name policy");
126  static_assert(!traits::has_short_name_method_v<basic_one_of_t>,
127  "basic_one_of_t must not have a short name policy");
128  static_assert(!traits::has_none_name_method_v<basic_one_of_t>,
129  "basic_one_of_t must not have a none name policy");
130  static_assert(!traits::has_description_method_v<basic_one_of_t>,
131  "basic_one_of_t must not have a description policy");
132 
133 protected:
134  using typename parent_type::children_type;
135  using typename parent_type::policies_type;
136 
137  static_assert(
138  boost::mp11::mp_any_of_q<
139  policies_type,
140  boost::mp11::mp_bind<policy::has_missing_phase_method, boost::mp11::_1, bool>>::value,
141  "basic_one_of_t must have a missing phase method, a "
142  "policy::required or policy::default_value are commonly used");
143 
144  static_assert(!parent_type::template any_phases_v<bool,
145  policy::has_pre_parse_phase_method,
146  policy::has_parse_phase_method,
147  policy::has_routing_phase_method>,
148  "basic_one_of_t does not support policies with pre-parse, parse, "
149  "or routing phases; as it delegates those to its children");
150 
151  using basic_value_type =
152  boost::mp11::mp_transform<traits::get_value_type,
153  boost::mp11::mp_remove_if<typename parent_type::children_type,
155 
156  static_assert((std::tuple_size_v<basic_value_type> >= 1),
157  "basic_one_of_t must have at least one child with a value_type");
158 
159  template <bool Flatten>
160  class children_help_data_type
161  {
162  template <typename Child, typename Prefix>
163  struct prefixer {
164  using label = typename Prefix::template append_t<
165  typename Child::template help_data_type<Flatten>::label>;
166  using description = typename Child::template help_data_type<Flatten>::description;
167  using children = std::tuple<>;
168  };
169 
170  using top_bar = AR_STRING("┌ ");
171  using middle_bar = AR_STRING("├ ");
172  using bottom_bar = AR_STRING("└ ");
173 
174  template <typename Child>
175  using first_prefixer = prefixer<Child, top_bar>;
176 
177  template <typename Child>
178  using middle_prefixer = prefixer<Child, middle_bar>;
179 
180  template <typename Child>
181  using last_prefixer = prefixer<Child, bottom_bar>;
182 
183  public:
184  using children = boost::mp11::mp_replace_at_c<
185  boost::mp11::mp_replace_first<boost::mp11::mp_transform<middle_prefixer, children_type>,
186  first_prefixer<boost::mp11::mp_first<children_type>>>,
187  std::tuple_size_v<children_type> - 1,
188  last_prefixer<boost::mp11::mp_back<children_type>>>;
189 
190  template <typename OwnerNode, typename FilterFn>
191  [[nodiscard]] static vector<runtime_help_data> runtime_children(const OwnerNode& owner,
192  FilterFn&& f)
193  {
194  // Gather the basic data
195  auto result = parent_type::template default_leaf_help_data_type<true>::runtime_children(
196  owner,
197  std::forward<FilterFn>(f));
198 
199  // Don't prepend bars if there are zero or one children
200  if (result.size() <= 1) {
201  return result;
202  }
203 
204  // Prepend the bars
205  result.front().label =
206  utility::dynamic_string_view{top_bar::get()} + result.front().label;
207  for (auto i = 1u; i < (result.size() - 1); ++i) {
208  result[i].label = utility::dynamic_string_view{middle_bar::get()} + result[i].label;
209  }
210  result.back().label =
211  utility::dynamic_string_view{bottom_bar::get()} + result.back().label;
212 
213  return result;
214  }
215  };
216 
217  template <auto has_display_name = add_names<ParentDocName, Params...>::has_display_name>
218  constexpr explicit basic_one_of_t(Params... params,
219  // NOLINTNEXTLINE(*-named-parameter)
220  std::enable_if_t<has_display_name>* = nullptr) noexcept :
221  parent_type{std::tuple_element_t<0, policies_type>{}, // Error name
222  std::move(params)...}
223  {
224  }
225 
226  template <auto has_display_name = add_names<ParentDocName, Params...>::has_display_name>
227  constexpr explicit basic_one_of_t(Params... params,
228  // NOLINTNEXTLINE(*-named-parameter)
229  std::enable_if_t<!has_display_name>* = nullptr) noexcept :
230  parent_type{std::tuple_element_t<0, policies_type>{}, // Display name
231  std::tuple_element_t<1, policies_type>{}, // Error name
232  std::move(params)...}
233  {
234  }
235 };
236 } // namespace arg_router::dependency::detail
constexpr std::string_view to_string(prefix_type prefix) noexcept
Definition: token_type.hpp:25
std::is_base_of< no_result_value<>, T > has_no_result_value
constexpr auto description
Definition: description.hpp:50
typename T::type get_type
Definition: traits.hpp:62
typename T::value_type get_value_type
Definition: traits.hpp:69
typename convert_to_cts< T >::type convert_to_cts_t