Line data Source code
1 : /*
2 : __ _ _ __ __ _ _ __ __ _ _ __ ___ ___
3 : / _` | '__/ _` | '_ \ / _` | '__/ __|/ _ \ Argument Parser for Modern C++
4 : | (_| | | | (_| | |_) | (_| | | \__ \ __/ http://github.com/p-ranav/argparse
5 : \__,_|_| \__, | .__/ \__,_|_| |___/\___|
6 : |___/|_|
7 :
8 : Licensed under the MIT License <http://opensource.org/licenses/MIT>.
9 : SPDX-License-Identifier: MIT
10 : Copyright (c) 2019-2022 Pranav Srinivas Kumar <pranav.srinivas.kumar@gmail.com>
11 : and other contributors.
12 :
13 : Permission is hereby granted, free of charge, to any person obtaining a copy
14 : of this software and associated documentation files (the "Software"), to deal
15 : in the Software without restriction, including without limitation the rights
16 : to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
17 : copies of the Software, and to permit persons to whom the Software is
18 : furnished to do so, subject to the following conditions:
19 :
20 : The above copyright notice and this permission notice shall be included in all
21 : copies or substantial portions of the Software.
22 :
23 : THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
24 : IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
25 : FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
26 : AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
27 : LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
28 : OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
29 : SOFTWARE.
30 : */
31 : #pragma once
32 :
33 : #include <cerrno>
34 :
35 : #ifndef ARGPARSE_MODULE_USE_STD_MODULE
36 : #include <algorithm>
37 : #include <any>
38 : #include <array>
39 : #include <set>
40 : #include <charconv>
41 : #include <cstdlib>
42 : #include <functional>
43 : #include <iomanip>
44 : #include <iostream>
45 : #include <iterator>
46 : #include <limits>
47 : #include <list>
48 : #include <map>
49 : #include <numeric>
50 : #include <optional>
51 : #include <sstream>
52 : #include <stdexcept>
53 : #include <string>
54 : #include <string_view>
55 : #include <tuple>
56 : #include <type_traits>
57 : #include <utility>
58 : #include <variant>
59 : #include <vector>
60 : #endif
61 :
62 : #ifndef ARGPARSE_CUSTOM_STRTOF
63 : #define ARGPARSE_CUSTOM_STRTOF strtof
64 : #endif
65 :
66 : #ifndef ARGPARSE_CUSTOM_STRTOD
67 : #define ARGPARSE_CUSTOM_STRTOD strtod
68 : #endif
69 :
70 : #ifndef ARGPARSE_CUSTOM_STRTOLD
71 : #define ARGPARSE_CUSTOM_STRTOLD strtold
72 : #endif
73 :
74 : namespace argparse {
75 :
76 : namespace details { // namespace for helper methods
77 :
78 : template <typename T, typename = void>
79 : struct HasContainerTraits : std::false_type {};
80 :
81 : template <> struct HasContainerTraits<std::string> : std::false_type {};
82 :
83 : template <> struct HasContainerTraits<std::string_view> : std::false_type {};
84 :
85 : template <typename T>
86 : struct HasContainerTraits<
87 : T, std::void_t<typename T::value_type, decltype(std::declval<T>().begin()),
88 : decltype(std::declval<T>().end()),
89 : decltype(std::declval<T>().size())>> : std::true_type {};
90 :
91 : template <typename T>
92 : inline constexpr bool IsContainer = HasContainerTraits<T>::value;
93 :
94 : template <typename T, typename = void>
95 : struct HasStreamableTraits : std::false_type {};
96 :
97 : template <typename T>
98 : struct HasStreamableTraits<
99 : T,
100 : std::void_t<decltype(std::declval<std::ostream &>() << std::declval<T>())>>
101 : : std::true_type {};
102 :
103 : template <typename T>
104 : inline constexpr bool IsStreamable = HasStreamableTraits<T>::value;
105 :
106 : constexpr std::size_t repr_max_container_size = 5;
107 :
108 96981 : template <typename T> std::string repr(T const &val) {
109 : if constexpr (std::is_same_v<T, bool>) {
110 95598 : return val ? "true" : "false";
111 : } else if constexpr (std::is_convertible_v<T, std::string_view>) {
112 194 : return '"' + std::string{std::string_view{val}} + '"';
113 : } else if constexpr (IsContainer<T>) {
114 : std::stringstream out;
115 : out << "{";
116 : const auto size = val.size();
117 : if (size > 1) {
118 : out << repr(*val.begin());
119 : std::for_each(
120 : std::next(val.begin()),
121 : std::next(
122 : val.begin(),
123 : static_cast<typename T::iterator::difference_type>(
124 : std::min<std::size_t>(size, repr_max_container_size) - 1)),
125 : [&out](const auto &v) { out << " " << repr(v); });
126 : if (size <= repr_max_container_size) {
127 : out << " ";
128 : } else {
129 : out << "...";
130 : }
131 : }
132 : if (size > 0) {
133 : out << repr(*std::prev(val.end()));
134 : }
135 : out << "}";
136 : return out.str();
137 : } else if constexpr (IsStreamable<T>) {
138 2378 : std::stringstream out;
139 1189 : out << val;
140 2378 : return out.str();
141 : } else {
142 : return "<not representable>";
143 : }
144 : }
145 :
146 : namespace {
147 :
148 : template <typename T> constexpr bool standard_signed_integer = false;
149 : template <> constexpr bool standard_signed_integer<signed char> = true;
150 : template <> constexpr bool standard_signed_integer<short int> = true;
151 : template <> constexpr bool standard_signed_integer<int> = true;
152 : template <> constexpr bool standard_signed_integer<long int> = true;
153 : template <> constexpr bool standard_signed_integer<long long int> = true;
154 :
155 : template <typename T> constexpr bool standard_unsigned_integer = false;
156 : template <> constexpr bool standard_unsigned_integer<unsigned char> = true;
157 : template <> constexpr bool standard_unsigned_integer<unsigned short int> = true;
158 : template <> constexpr bool standard_unsigned_integer<unsigned int> = true;
159 : template <> constexpr bool standard_unsigned_integer<unsigned long int> = true;
160 : template <>
161 : constexpr bool standard_unsigned_integer<unsigned long long int> = true;
162 :
163 : } // namespace
164 :
165 : constexpr int radix_2 = 2;
166 : constexpr int radix_8 = 8;
167 : constexpr int radix_10 = 10;
168 : constexpr int radix_16 = 16;
169 :
170 : template <typename T>
171 : constexpr bool standard_integer =
172 : standard_signed_integer<T> || standard_unsigned_integer<T>;
173 :
174 : template <class F, class Tuple, class Extra, std::size_t... I>
175 : constexpr decltype(auto)
176 : apply_plus_one_impl(F &&f, Tuple &&t, Extra &&x,
177 : std::index_sequence<I...> /*unused*/) {
178 : return std::invoke(std::forward<F>(f), std::get<I>(std::forward<Tuple>(t))...,
179 : std::forward<Extra>(x));
180 : }
181 :
182 : template <class F, class Tuple, class Extra>
183 : constexpr decltype(auto) apply_plus_one(F &&f, Tuple &&t, Extra &&x) {
184 : return details::apply_plus_one_impl(
185 : std::forward<F>(f), std::forward<Tuple>(t), std::forward<Extra>(x),
186 : std::make_index_sequence<
187 : std::tuple_size_v<std::remove_reference_t<Tuple>>>{});
188 : }
189 :
190 6665 : constexpr auto pointer_range(std::string_view s) noexcept {
191 6665 : return std::tuple(s.data(), s.data() + s.size());
192 : }
193 :
194 : template <class CharT, class Traits>
195 25540 : constexpr bool starts_with(std::basic_string_view<CharT, Traits> prefix,
196 : std::basic_string_view<CharT, Traits> s) noexcept {
197 25540 : return s.substr(0, prefix.size()) == prefix;
198 : }
199 :
200 : enum class chars_format {
201 : scientific = 0xf1,
202 : fixed = 0xf2,
203 : hex = 0xf4,
204 : binary = 0xf8,
205 : general = fixed | scientific
206 : };
207 :
208 : struct ConsumeBinaryPrefixResult {
209 : bool is_binary;
210 : std::string_view rest;
211 : };
212 :
213 6244 : constexpr auto consume_binary_prefix(std::string_view s)
214 : -> ConsumeBinaryPrefixResult {
215 12488 : if (starts_with(std::string_view{"0b"}, s) ||
216 12488 : starts_with(std::string_view{"0B"}, s)) {
217 0 : s.remove_prefix(2);
218 0 : return {true, s};
219 : }
220 6244 : return {false, s};
221 : }
222 :
223 : struct ConsumeHexPrefixResult {
224 : bool is_hexadecimal;
225 : std::string_view rest;
226 : };
227 :
228 : using namespace std::literals;
229 :
230 6244 : constexpr auto consume_hex_prefix(std::string_view s)
231 : -> ConsumeHexPrefixResult {
232 6244 : if (starts_with("0x"sv, s) || starts_with("0X"sv, s)) {
233 0 : s.remove_prefix(2);
234 0 : return {true, s};
235 : }
236 6244 : return {false, s};
237 : }
238 :
239 : template <class T, auto Param>
240 985 : inline auto do_from_chars(std::string_view s) -> T {
241 985 : T x{0};
242 985 : auto [first, last] = pointer_range(s);
243 985 : auto [ptr, ec] = std::from_chars(first, last, x, Param);
244 985 : if (ec == std::errc()) {
245 985 : if (ptr == last) {
246 985 : return x;
247 : }
248 : throw std::invalid_argument{"pattern '" + std::string(s) +
249 0 : "' does not match to the end"};
250 : }
251 0 : if (ec == std::errc::invalid_argument) {
252 0 : throw std::invalid_argument{"pattern '" + std::string(s) + "' not found"};
253 : }
254 0 : if (ec == std::errc::result_out_of_range) {
255 0 : throw std::range_error{"'" + std::string(s) + "' not representable"};
256 : }
257 0 : return x; // unreachable
258 : }
259 :
260 : template <class T, auto Param = 0> struct parse_number {
261 421 : auto operator()(std::string_view s) -> T {
262 421 : return do_from_chars<T, Param>(s);
263 : }
264 : };
265 :
266 : template <class T> struct parse_number<T, radix_2> {
267 : auto operator()(std::string_view s) -> T {
268 : if (auto [ok, rest] = consume_binary_prefix(s); ok) {
269 : return do_from_chars<T, radix_2>(rest);
270 : }
271 : throw std::invalid_argument{"pattern not found"};
272 : }
273 : };
274 :
275 : template <class T> struct parse_number<T, radix_16> {
276 : auto operator()(std::string_view s) -> T {
277 : if (starts_with("0x"sv, s) || starts_with("0X"sv, s)) {
278 : if (auto [ok, rest] = consume_hex_prefix(s); ok) {
279 : try {
280 : return do_from_chars<T, radix_16>(rest);
281 : } catch (const std::invalid_argument &err) {
282 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
283 : "' as hexadecimal: " + err.what());
284 : } catch (const std::range_error &err) {
285 : throw std::range_error("Failed to parse '" + std::string(s) +
286 : "' as hexadecimal: " + err.what());
287 : }
288 : }
289 : } else {
290 : // Allow passing hex numbers without prefix
291 : // Shape 'x' already has to be specified
292 : try {
293 : return do_from_chars<T, radix_16>(s);
294 : } catch (const std::invalid_argument &err) {
295 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
296 : "' as hexadecimal: " + err.what());
297 : } catch (const std::range_error &err) {
298 : throw std::range_error("Failed to parse '" + std::string(s) +
299 : "' as hexadecimal: " + err.what());
300 : }
301 : }
302 :
303 : throw std::invalid_argument{"pattern '" + std::string(s) +
304 : "' not identified as hexadecimal"};
305 : }
306 : };
307 :
308 : template <class T> struct parse_number<T> {
309 564 : auto operator()(std::string_view s) -> T {
310 564 : auto [ok, rest] = consume_hex_prefix(s);
311 564 : if (ok) {
312 : try {
313 0 : return do_from_chars<T, radix_16>(rest);
314 0 : } catch (const std::invalid_argument &err) {
315 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
316 0 : "' as hexadecimal: " + err.what());
317 0 : } catch (const std::range_error &err) {
318 : throw std::range_error("Failed to parse '" + std::string(s) +
319 0 : "' as hexadecimal: " + err.what());
320 : }
321 : }
322 :
323 564 : auto [ok_binary, rest_binary] = consume_binary_prefix(s);
324 564 : if (ok_binary) {
325 : try {
326 0 : return do_from_chars<T, radix_2>(rest_binary);
327 0 : } catch (const std::invalid_argument &err) {
328 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
329 0 : "' as binary: " + err.what());
330 0 : } catch (const std::range_error &err) {
331 : throw std::range_error("Failed to parse '" + std::string(s) +
332 0 : "' as binary: " + err.what());
333 : }
334 : }
335 :
336 564 : if (starts_with("0"sv, s)) {
337 : try {
338 10 : return do_from_chars<T, radix_8>(rest);
339 0 : } catch (const std::invalid_argument &err) {
340 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
341 0 : "' as octal: " + err.what());
342 0 : } catch (const std::range_error &err) {
343 : throw std::range_error("Failed to parse '" + std::string(s) +
344 0 : "' as octal: " + err.what());
345 : }
346 : }
347 :
348 : try {
349 554 : return do_from_chars<T, radix_10>(rest);
350 0 : } catch (const std::invalid_argument &err) {
351 : throw std::invalid_argument("Failed to parse '" + std::string(s) +
352 0 : "' as decimal integer: " + err.what());
353 0 : } catch (const std::range_error &err) {
354 : throw std::range_error("Failed to parse '" + std::string(s) +
355 0 : "' as decimal integer: " + err.what());
356 : }
357 : }
358 : };
359 :
360 : namespace {
361 :
362 : template <class T> inline const auto generic_strtod = nullptr;
363 : template <> inline const auto generic_strtod<float> = ARGPARSE_CUSTOM_STRTOF;
364 : template <> inline const auto generic_strtod<double> = ARGPARSE_CUSTOM_STRTOD;
365 : template <>
366 : inline const auto generic_strtod<long double> = ARGPARSE_CUSTOM_STRTOLD;
367 :
368 : } // namespace
369 :
370 5680 : template <class T> inline auto do_strtod(std::string const &s) -> T {
371 5680 : if (isspace(static_cast<unsigned char>(s[0])) || s[0] == '+') {
372 0 : throw std::invalid_argument{"pattern '" + s + "' not found"};
373 : }
374 :
375 5680 : auto [first, last] = pointer_range(s);
376 : char *ptr;
377 :
378 5680 : errno = 0;
379 5680 : auto x = generic_strtod<T>(first, &ptr);
380 5680 : if (errno == 0) {
381 5680 : if (ptr == last) {
382 5675 : return x;
383 : }
384 : throw std::invalid_argument{"pattern '" + s +
385 5 : "' does not match to the end"};
386 : }
387 0 : if (errno == ERANGE) {
388 0 : throw std::range_error{"'" + s + "' not representable"};
389 : }
390 0 : return x; // unreachable
391 : }
392 :
393 : template <class T> struct parse_number<T, chars_format::general> {
394 5680 : auto operator()(std::string const &s) -> T {
395 5680 : if (auto r = consume_hex_prefix(s); r.is_hexadecimal) {
396 : throw std::invalid_argument{
397 0 : "chars_format::general does not parse hexfloat"};
398 : }
399 5680 : if (auto r = consume_binary_prefix(s); r.is_binary) {
400 : throw std::invalid_argument{
401 0 : "chars_format::general does not parse binfloat"};
402 : }
403 :
404 : try {
405 5680 : return do_strtod<T>(s);
406 10 : } catch (const std::invalid_argument &err) {
407 : throw std::invalid_argument("Failed to parse '" + s +
408 5 : "' as number: " + err.what());
409 0 : } catch (const std::range_error &err) {
410 : throw std::range_error("Failed to parse '" + s +
411 0 : "' as number: " + err.what());
412 : }
413 : }
414 : };
415 :
416 : template <class T> struct parse_number<T, chars_format::hex> {
417 : auto operator()(std::string const &s) -> T {
418 : if (auto r = consume_hex_prefix(s); !r.is_hexadecimal) {
419 : throw std::invalid_argument{"chars_format::hex parses hexfloat"};
420 : }
421 : if (auto r = consume_binary_prefix(s); r.is_binary) {
422 : throw std::invalid_argument{"chars_format::hex does not parse binfloat"};
423 : }
424 :
425 : try {
426 : return do_strtod<T>(s);
427 : } catch (const std::invalid_argument &err) {
428 : throw std::invalid_argument("Failed to parse '" + s +
429 : "' as hexadecimal: " + err.what());
430 : } catch (const std::range_error &err) {
431 : throw std::range_error("Failed to parse '" + s +
432 : "' as hexadecimal: " + err.what());
433 : }
434 : }
435 : };
436 :
437 : template <class T> struct parse_number<T, chars_format::binary> {
438 : auto operator()(std::string const &s) -> T {
439 : if (auto r = consume_hex_prefix(s); r.is_hexadecimal) {
440 : throw std::invalid_argument{
441 : "chars_format::binary does not parse hexfloat"};
442 : }
443 : if (auto r = consume_binary_prefix(s); !r.is_binary) {
444 : throw std::invalid_argument{"chars_format::binary parses binfloat"};
445 : }
446 :
447 : return do_strtod<T>(s);
448 : }
449 : };
450 :
451 : template <class T> struct parse_number<T, chars_format::scientific> {
452 : auto operator()(std::string const &s) -> T {
453 : if (auto r = consume_hex_prefix(s); r.is_hexadecimal) {
454 : throw std::invalid_argument{
455 : "chars_format::scientific does not parse hexfloat"};
456 : }
457 : if (auto r = consume_binary_prefix(s); r.is_binary) {
458 : throw std::invalid_argument{
459 : "chars_format::scientific does not parse binfloat"};
460 : }
461 : if (s.find_first_of("eE") == std::string::npos) {
462 : throw std::invalid_argument{
463 : "chars_format::scientific requires exponent part"};
464 : }
465 :
466 : try {
467 : return do_strtod<T>(s);
468 : } catch (const std::invalid_argument &err) {
469 : throw std::invalid_argument("Failed to parse '" + s +
470 : "' as scientific notation: " + err.what());
471 : } catch (const std::range_error &err) {
472 : throw std::range_error("Failed to parse '" + s +
473 : "' as scientific notation: " + err.what());
474 : }
475 : }
476 : };
477 :
478 : template <class T> struct parse_number<T, chars_format::fixed> {
479 : auto operator()(std::string const &s) -> T {
480 : if (auto r = consume_hex_prefix(s); r.is_hexadecimal) {
481 : throw std::invalid_argument{
482 : "chars_format::fixed does not parse hexfloat"};
483 : }
484 : if (auto r = consume_binary_prefix(s); r.is_binary) {
485 : throw std::invalid_argument{
486 : "chars_format::fixed does not parse binfloat"};
487 : }
488 : if (s.find_first_of("eE") != std::string::npos) {
489 : throw std::invalid_argument{
490 : "chars_format::fixed does not parse exponent part"};
491 : }
492 :
493 : try {
494 : return do_strtod<T>(s);
495 : } catch (const std::invalid_argument &err) {
496 : throw std::invalid_argument("Failed to parse '" + s +
497 : "' as fixed notation: " + err.what());
498 : } catch (const std::range_error &err) {
499 : throw std::range_error("Failed to parse '" + s +
500 : "' as fixed notation: " + err.what());
501 : }
502 : }
503 : };
504 :
505 : template <typename StrIt>
506 0 : std::string join(StrIt first, StrIt last, const std::string &separator) {
507 0 : if (first == last) {
508 0 : return "";
509 : }
510 0 : std::stringstream value;
511 0 : value << *first;
512 0 : ++first;
513 0 : while (first != last) {
514 0 : value << separator << *first;
515 0 : ++first;
516 : }
517 0 : return value.str();
518 : }
519 :
520 : template <typename T> struct can_invoke_to_string {
521 : template <typename U>
522 : static auto test(int)
523 : -> decltype(std::to_string(std::declval<U>()), std::true_type{});
524 :
525 : template <typename U> static auto test(...) -> std::false_type;
526 :
527 : static constexpr bool value = decltype(test<T>(0))::value;
528 : };
529 :
530 : template <typename T> struct IsChoiceTypeSupported {
531 : using CleanType = typename std::decay<T>::type;
532 : static const bool value = std::is_integral<CleanType>::value ||
533 : std::is_same<CleanType, std::string>::value ||
534 : std::is_same<CleanType, std::string_view>::value ||
535 : std::is_same<CleanType, const char *>::value;
536 : };
537 :
538 : template <typename StringType>
539 0 : std::size_t get_levenshtein_distance(const StringType &s1,
540 : const StringType &s2) {
541 0 : std::vector<std::vector<std::size_t>> dp(
542 0 : s1.size() + 1, std::vector<std::size_t>(s2.size() + 1, 0));
543 :
544 0 : for (std::size_t i = 0; i <= s1.size(); ++i) {
545 0 : for (std::size_t j = 0; j <= s2.size(); ++j) {
546 0 : if (i == 0) {
547 0 : dp[i][j] = j;
548 0 : } else if (j == 0) {
549 0 : dp[i][j] = i;
550 0 : } else if (s1[i - 1] == s2[j - 1]) {
551 0 : dp[i][j] = dp[i - 1][j - 1];
552 : } else {
553 0 : dp[i][j] = 1 + std::min({dp[i - 1][j], dp[i][j - 1], dp[i - 1][j - 1]});
554 : }
555 : }
556 : }
557 :
558 0 : return dp[s1.size()][s2.size()];
559 : }
560 :
561 : template <typename ValueType>
562 0 : std::string get_most_similar_string(const std::map<std::string, ValueType> &map,
563 : const std::string &input) {
564 0 : std::string most_similar{};
565 0 : std::size_t min_distance = std::numeric_limits<std::size_t>::max();
566 :
567 0 : for (const auto &entry : map) {
568 0 : std::size_t distance = get_levenshtein_distance(entry.first, input);
569 0 : if (distance < min_distance) {
570 0 : min_distance = distance;
571 0 : most_similar = entry.first;
572 : }
573 : }
574 :
575 0 : return most_similar;
576 : }
577 :
578 : } // namespace details
579 :
580 : enum class nargs_pattern { optional, any, at_least_one };
581 :
582 : enum class default_arguments : unsigned int {
583 : none = 0,
584 : help = 1,
585 : version = 2,
586 : all = help | version,
587 : };
588 :
589 11202 : inline default_arguments operator&(const default_arguments &a,
590 : const default_arguments &b) {
591 : return static_cast<default_arguments>(
592 11202 : static_cast<std::underlying_type<default_arguments>::type>(a) &
593 11202 : static_cast<std::underlying_type<default_arguments>::type>(b));
594 : }
595 :
596 : class ArgumentParser;
597 :
598 : class Argument {
599 : friend class ArgumentParser;
600 : friend auto operator<<(std::ostream &stream, const ArgumentParser &parser)
601 : -> std::ostream &;
602 :
603 : template <std::size_t N, std::size_t... I>
604 243420 : explicit Argument(std::string_view prefix_chars,
605 : std::array<std::string_view, N> &&a,
606 : std::index_sequence<I...> /*unused*/)
607 : : m_accepts_optional_like_value(false),
608 243420 : m_is_optional((is_optional(a[I], prefix_chars) || ...)),
609 : m_is_required(false), m_is_repeatable(false), m_is_used(false),
610 251858 : m_is_hidden(false), m_prefix_chars(prefix_chars) {
611 243420 : ((void)m_names.emplace_back(a[I]), ...);
612 243420 : std::sort(
613 16876 : m_names.begin(), m_names.end(), [](const auto &lhs, const auto &rhs) {
614 16876 : return lhs.size() == rhs.size() ? lhs < rhs : lhs.size() < rhs.size();
615 : });
616 243420 : }
617 :
618 : public:
619 : template <std::size_t N>
620 243420 : explicit Argument(std::string_view prefix_chars,
621 : std::array<std::string_view, N> &&a)
622 243420 : : Argument(prefix_chars, std::move(a), std::make_index_sequence<N>{}) {}
623 :
624 237010 : Argument &help(std::string help_text) {
625 237010 : m_help = std::move(help_text);
626 237010 : return *this;
627 : }
628 :
629 156878 : Argument &metavar(std::string metavar) {
630 156878 : m_metavar = std::move(metavar);
631 156878 : return *this;
632 : }
633 :
634 96981 : template <typename T> Argument &default_value(T &&value) {
635 96981 : m_num_args_range = NArgsRange{0, m_num_args_range.get_max()};
636 96981 : m_default_value_repr = details::repr(value);
637 :
638 : if constexpr (std::is_convertible_v<T, std::string_view>) {
639 194 : m_default_value_str = std::string{std::string_view{value}};
640 : } else if constexpr (details::can_invoke_to_string<T>::value) {
641 96787 : m_default_value_str = std::to_string(value);
642 : }
643 :
644 96981 : m_default_value = std::forward<T>(value);
645 96981 : return *this;
646 : }
647 :
648 194 : Argument &default_value(const char *value) {
649 194 : return default_value(std::string(value));
650 : }
651 :
652 : Argument &required() {
653 : m_is_required = true;
654 : return *this;
655 : }
656 :
657 95598 : Argument &implicit_value(std::any value) {
658 95598 : m_implicit_value = std::move(value);
659 95598 : m_num_args_range = NArgsRange{0, 0};
660 95598 : return *this;
661 : }
662 :
663 : // This is shorthand for:
664 : // program.add_argument("foo")
665 : // .default_value(false)
666 : // .implicit_value(true)
667 94567 : Argument &flag() {
668 94567 : default_value(false);
669 94567 : implicit_value(true);
670 94567 : return *this;
671 : }
672 :
673 : template <class F, class... Args>
674 229103 : auto action(F &&callable, Args &&... bound_args)
675 : -> std::enable_if_t<std::is_invocable_v<F, Args..., std::string const>,
676 : Argument &> {
677 : using action_type = std::conditional_t<
678 : std::is_void_v<std::invoke_result_t<F, Args..., std::string const>>,
679 : void_action, valued_action>;
680 : if constexpr (sizeof...(Args) == 0) {
681 229103 : m_actions.emplace_back<action_type>(std::forward<F>(callable));
682 : } else {
683 : m_actions.emplace_back<action_type>(
684 : [f = std::forward<F>(callable),
685 : tup = std::make_tuple(std::forward<Args>(bound_args)...)](
686 : std::string const &opt) mutable {
687 : return details::apply_plus_one(f, tup, opt);
688 : });
689 : }
690 229103 : return *this;
691 : }
692 :
693 40937 : auto &store_into(bool &var) {
694 40937 : flag();
695 40937 : if (m_default_value.has_value()) {
696 40937 : var = std::any_cast<bool>(m_default_value);
697 : }
698 40937 : action([&var](const auto & /*unused*/) { var = true; });
699 40937 : return *this;
700 : }
701 :
702 : template <typename T, typename std::enable_if<std::is_integral<T>::value>::type * = nullptr>
703 6093 : auto &store_into(T &var) {
704 6093 : if (m_default_value.has_value()) {
705 : try
706 : {
707 214 : var = std::any_cast<T>(m_default_value);
708 : }
709 21 : catch (...)
710 : {
711 21 : var = static_cast<T>(std::any_cast<int>(m_default_value));
712 : }
713 : }
714 6174 : action([&var](const auto &s) {
715 147 : var = details::parse_number<T, details::radix_10>()(s);
716 : });
717 6093 : return *this;
718 : }
719 :
720 14552 : auto &store_into(double &var) {
721 14552 : if (m_default_value.has_value()) {
722 : try
723 : {
724 975 : var = std::any_cast<double>(m_default_value);
725 : }
726 105 : catch (...)
727 : {
728 105 : var = std::any_cast<int>(m_default_value);
729 : }
730 : }
731 224 : action([&var](const auto &s) {
732 224 : var = details::parse_number<double, details::chars_format::general>()(s);
733 14771 : });
734 14552 : return *this;
735 : }
736 :
737 31435 : auto &store_into(std::string &var) {
738 31435 : if (m_default_value.has_value()) {
739 50 : var = std::any_cast<std::string>(m_default_value);
740 : }
741 35160 : action([&var](const std::string &s) { var = s; });
742 31435 : return *this;
743 : }
744 :
745 430 : auto &store_into(std::vector<std::string> &var) {
746 430 : if (m_default_value.has_value()) {
747 0 : var = std::any_cast<std::vector<std::string>>(m_default_value);
748 : }
749 602 : action([this, &var](const std::string &s) {
750 202 : if (!m_is_used) {
751 198 : var.clear();
752 : }
753 202 : m_is_used = true;
754 202 : var.push_back(s);
755 430 : });
756 430 : return *this;
757 : }
758 :
759 2043 : auto &store_into(std::vector<int> &var) {
760 2043 : if (m_default_value.has_value()) {
761 0 : var = std::any_cast<std::vector<int>>(m_default_value);
762 : }
763 593 : action([this, &var](const std::string &s) {
764 274 : if (!m_is_used) {
765 45 : var.clear();
766 : }
767 274 : m_is_used = true;
768 274 : var.push_back(details::parse_number<int, details::radix_10>()(s));
769 2317 : });
770 2043 : return *this;
771 : }
772 :
773 29 : auto &store_into(std::set<std::string> &var) {
774 29 : if (m_default_value.has_value()) {
775 0 : var = std::any_cast<std::set<std::string>>(m_default_value);
776 : }
777 6 : action([this, &var](const std::string &s) {
778 2 : if (!m_is_used) {
779 2 : var.clear();
780 : }
781 2 : m_is_used = true;
782 2 : var.insert(s);
783 29 : });
784 29 : return *this;
785 : }
786 :
787 : auto &store_into(std::set<int> &var) {
788 : if (m_default_value.has_value()) {
789 : var = std::any_cast<std::set<int>>(m_default_value);
790 : }
791 : action([this, &var](const std::string &s) {
792 : if (!m_is_used) {
793 : var.clear();
794 : }
795 : m_is_used = true;
796 : var.insert(details::parse_number<int, details::radix_10>()(s));
797 : });
798 : return *this;
799 : }
800 :
801 42005 : auto &append() {
802 42005 : m_is_repeatable = true;
803 42005 : return *this;
804 : }
805 :
806 : // Cause the argument to be invisible in usage and help
807 15473 : auto &hidden() {
808 15473 : m_is_hidden = true;
809 15473 : return *this;
810 : }
811 :
812 : template <char Shape, typename T>
813 29844 : auto scan() -> std::enable_if_t<std::is_arithmetic_v<T>, Argument &> {
814 : static_assert(!(std::is_const_v<T> || std::is_volatile_v<T>),
815 : "T should not be cv-qualified");
816 : auto is_one_of = [](char c, auto... x) constexpr {
817 : return ((c == x) || ...);
818 : };
819 :
820 : if constexpr (is_one_of(Shape, 'd') && details::standard_integer<T>) {
821 : action(details::parse_number<T, details::radix_10>());
822 : } else if constexpr (is_one_of(Shape, 'i') &&
823 : details::standard_integer<T>) {
824 2519 : action(details::parse_number<T>());
825 : } else if constexpr (is_one_of(Shape, 'u') &&
826 : details::standard_unsigned_integer<T>) {
827 : action(details::parse_number<T, details::radix_10>());
828 : } else if constexpr (is_one_of(Shape, 'b') &&
829 : details::standard_unsigned_integer<T>) {
830 : action(details::parse_number<T, details::radix_2>());
831 : } else if constexpr (is_one_of(Shape, 'o') &&
832 : details::standard_unsigned_integer<T>) {
833 : action(details::parse_number<T, details::radix_8>());
834 : } else if constexpr (is_one_of(Shape, 'x', 'X') &&
835 : details::standard_unsigned_integer<T>) {
836 : action(details::parse_number<T, details::radix_16>());
837 : } else if constexpr (is_one_of(Shape, 'a', 'A') &&
838 : std::is_floating_point_v<T>) {
839 : action(details::parse_number<T, details::chars_format::hex>());
840 : } else if constexpr (is_one_of(Shape, 'e', 'E') &&
841 : std::is_floating_point_v<T>) {
842 : action(details::parse_number<T, details::chars_format::scientific>());
843 : } else if constexpr (is_one_of(Shape, 'f', 'F') &&
844 : std::is_floating_point_v<T>) {
845 : action(details::parse_number<T, details::chars_format::fixed>());
846 : } else if constexpr (is_one_of(Shape, 'g', 'G') &&
847 : std::is_floating_point_v<T>) {
848 27325 : action(details::parse_number<T, details::chars_format::general>());
849 : } else {
850 : static_assert(alignof(T) == 0, "No scan specification for T");
851 : }
852 :
853 29844 : return *this;
854 : }
855 :
856 19817 : Argument &nargs(std::size_t num_args) {
857 19817 : m_num_args_range = NArgsRange{num_args, num_args};
858 19817 : return *this;
859 : }
860 :
861 3096 : Argument &nargs(std::size_t num_args_min, std::size_t num_args_max) {
862 3096 : m_num_args_range = NArgsRange{num_args_min, num_args_max};
863 3096 : return *this;
864 : }
865 :
866 1231 : Argument &nargs(nargs_pattern pattern) {
867 1231 : switch (pattern) {
868 119 : case nargs_pattern::optional:
869 119 : m_num_args_range = NArgsRange{0, 1};
870 119 : break;
871 1000 : case nargs_pattern::any:
872 1000 : m_num_args_range =
873 1000 : NArgsRange{0, (std::numeric_limits<std::size_t>::max)()};
874 1000 : break;
875 112 : case nargs_pattern::at_least_one:
876 112 : m_num_args_range =
877 112 : NArgsRange{1, (std::numeric_limits<std::size_t>::max)()};
878 112 : break;
879 : }
880 1231 : return *this;
881 : }
882 :
883 972 : Argument &remaining() {
884 972 : m_accepts_optional_like_value = true;
885 972 : return nargs(nargs_pattern::any);
886 : }
887 :
888 4486 : template <typename T> void add_choice(T &&choice) {
889 : static_assert(details::IsChoiceTypeSupported<T>::value,
890 : "Only string or integer type supported for choice");
891 : static_assert(std::is_convertible_v<T, std::string_view> ||
892 : details::can_invoke_to_string<T>::value,
893 : "Choice is not convertible to string_type");
894 4486 : if (!m_choices.has_value()) {
895 1413 : m_choices = std::vector<std::string>{};
896 : }
897 :
898 : if constexpr (std::is_convertible_v<T, std::string_view>) {
899 8972 : m_choices.value().push_back(
900 4486 : std::string{std::string_view{std::forward<T>(choice)}});
901 : } else if constexpr (details::can_invoke_to_string<T>::value) {
902 : m_choices.value().push_back(std::to_string(std::forward<T>(choice)));
903 : }
904 4486 : }
905 :
906 1413 : Argument &choices() {
907 1413 : if (!m_choices.has_value()) {
908 0 : throw std::runtime_error("Zero choices provided");
909 : }
910 1413 : return *this;
911 : }
912 :
913 : template <typename T, typename... U>
914 4486 : Argument &choices(T &&first, U &&... rest) {
915 4486 : add_choice(std::forward<T>(first));
916 4486 : choices(std::forward<U>(rest)...);
917 4486 : return *this;
918 : }
919 :
920 1403 : void find_default_value_in_choices_or_throw() const {
921 :
922 1403 : const auto &choices = m_choices.value();
923 :
924 1403 : if (m_default_value.has_value()) {
925 96 : if (std::find(choices.begin(), choices.end(), m_default_value_str) ==
926 192 : choices.end()) {
927 : // provided arg not in list of allowed choices
928 : // report error
929 :
930 : std::string choices_as_csv =
931 0 : std::accumulate(choices.begin(), choices.end(), std::string(),
932 0 : [](const std::string &a, const std::string &b) {
933 0 : return a + (a.empty() ? "" : ", ") + b;
934 0 : });
935 :
936 : throw std::runtime_error(
937 0 : std::string{"Invalid default value "} + m_default_value_repr +
938 0 : " - allowed options: {" + choices_as_csv + "}");
939 : }
940 : }
941 1403 : }
942 :
943 : template <typename Iterator>
944 180 : void find_value_in_choices_or_throw(Iterator it) const {
945 :
946 180 : const auto &choices = m_choices.value();
947 :
948 180 : if (std::find(choices.begin(), choices.end(), *it) == choices.end()) {
949 : // provided arg not in list of allowed choices
950 : // report error
951 :
952 0 : std::string choices_as_csv =
953 : std::accumulate(choices.begin(), choices.end(), std::string(),
954 0 : [](const std::string &a, const std::string &b) {
955 0 : return a + (a.empty() ? "" : ", ") + b;
956 : });
957 :
958 : throw std::runtime_error(std::string{"Invalid argument "} +
959 0 : details::repr(*it) + " - allowed options: {" +
960 0 : choices_as_csv + "}");
961 : }
962 180 : }
963 :
964 : /* The dry_run parameter can be set to true to avoid running the actions,
965 : * and setting m_is_used. This may be used by a pre-processing step to do
966 : * a first iteration over arguments.
967 : */
968 : template <typename Iterator>
969 24486 : Iterator consume(Iterator start, Iterator end,
970 : std::string_view used_name = {}, bool dry_run = false) {
971 24486 : if (!m_is_repeatable && m_is_used) {
972 : throw std::runtime_error(
973 0 : std::string("Duplicate argument ").append(used_name));
974 : }
975 24486 : m_used_name = used_name;
976 :
977 24486 : if (m_choices.has_value()) {
978 : // Check each value in (start, end) and make sure
979 : // it is in the list of allowed choices/options
980 180 : std::size_t i = 0;
981 180 : auto max_number_of_args = m_num_args_range.get_max();
982 360 : for (auto it = start; it != end; ++it) {
983 257 : if (i == max_number_of_args) {
984 77 : break;
985 : }
986 180 : find_value_in_choices_or_throw(it);
987 180 : i += 1;
988 : }
989 : }
990 :
991 24486 : const auto num_args_max = m_num_args_range.get_max();
992 24486 : const auto num_args_min = m_num_args_range.get_min();
993 24486 : std::size_t dist = 0;
994 24486 : if (num_args_max == 0) {
995 2128 : if (!dry_run) {
996 968 : m_values.emplace_back(m_implicit_value);
997 1942 : for(auto &action: m_actions) {
998 1948 : std::visit([&](const auto &f) { f({}); }, action);
999 : }
1000 968 : if(m_actions.empty()){
1001 0 : std::visit([&](const auto &f) { f({}); }, m_default_action);
1002 : }
1003 968 : m_is_used = true;
1004 : }
1005 2128 : return start;
1006 : }
1007 22358 : if ((dist = static_cast<std::size_t>(std::distance(start, end))) >=
1008 : num_args_min) {
1009 22357 : if (num_args_max < dist) {
1010 12860 : end = std::next(start, static_cast<typename Iterator::difference_type>(
1011 : num_args_max));
1012 : }
1013 22357 : if (!m_accepts_optional_like_value) {
1014 22228 : end = std::find_if(
1015 : start, end,
1016 22228 : std::bind(is_optional, std::placeholders::_1, m_prefix_chars));
1017 22228 : dist = static_cast<std::size_t>(std::distance(start, end));
1018 22228 : if (dist < num_args_min) {
1019 : throw std::runtime_error("Too few arguments for '" +
1020 0 : std::string(m_used_name) + "'.");
1021 : }
1022 : }
1023 :
1024 : struct ActionApply {
1025 2162 : void operator()(valued_action &f) {
1026 2162 : std::transform(first, last, std::back_inserter(self.m_values), f);
1027 2162 : }
1028 :
1029 8456 : void operator()(void_action &f) {
1030 8456 : std::for_each(first, last, f);
1031 8425 : if (!self.m_default_value.has_value()) {
1032 8273 : if (!self.m_accepts_optional_like_value) {
1033 16528 : self.m_values.resize(
1034 8264 : static_cast<std::size_t>(std::distance(first, last)));
1035 : }
1036 : }
1037 8425 : }
1038 :
1039 : Iterator first, last;
1040 : Argument &self;
1041 : };
1042 22357 : if (!dry_run) {
1043 20744 : for(auto &action: m_actions) {
1044 10340 : std::visit(ActionApply{start, end, *this}, action);
1045 : }
1046 10404 : if(m_actions.empty()){
1047 278 : std::visit(ActionApply{start, end, *this}, m_default_action);
1048 : }
1049 10404 : m_is_used = true;
1050 : }
1051 22326 : return end;
1052 : }
1053 1 : if (m_default_value.has_value()) {
1054 0 : if (!dry_run) {
1055 0 : m_is_used = true;
1056 : }
1057 0 : return start;
1058 : }
1059 : throw std::runtime_error("Too few arguments for '" +
1060 1 : std::string(m_used_name) + "'.");
1061 : }
1062 :
1063 : /*
1064 : * @throws std::runtime_error if argument values are not valid
1065 : */
1066 250289 : void validate() const {
1067 250289 : if (m_is_optional) {
1068 : // TODO: check if an implicit value was programmed for this argument
1069 247930 : if (!m_is_used && !m_default_value.has_value() && m_is_required) {
1070 0 : throw_required_arg_not_used_error();
1071 : }
1072 247930 : if (m_is_used && m_is_required && m_values.empty()) {
1073 0 : throw_required_arg_no_value_provided_error();
1074 : }
1075 : } else {
1076 2364 : if (!m_num_args_range.contains(m_values.size()) &&
1077 5 : !m_default_value.has_value()) {
1078 5 : throw_nargs_range_validation_error();
1079 : }
1080 : }
1081 :
1082 250284 : if (m_choices.has_value()) {
1083 : // Make sure the default value (if provided)
1084 : // is in the list of choices
1085 1403 : find_default_value_in_choices_or_throw();
1086 : }
1087 250284 : }
1088 :
1089 2 : std::string get_names_csv(char separator = ',') const {
1090 : return std::accumulate(
1091 4 : m_names.begin(), m_names.end(), std::string{""},
1092 2 : [&](const std::string &result, const std::string &name) {
1093 4 : return result.empty() ? name : result + separator + name;
1094 6 : });
1095 : }
1096 :
1097 2 : std::string get_usage_full() const {
1098 4 : std::stringstream usage;
1099 :
1100 2 : usage << get_names_csv('/');
1101 4 : const std::string metavar = !m_metavar.empty() ? m_metavar : "VAR";
1102 2 : if (m_num_args_range.get_max() > 0) {
1103 2 : usage << " " << metavar;
1104 2 : if (m_num_args_range.get_max() > 1) {
1105 0 : usage << "...";
1106 : }
1107 : }
1108 4 : return usage.str();
1109 : }
1110 :
1111 558 : std::string get_inline_usage() const {
1112 1116 : std::stringstream usage;
1113 : // Find the longest variant to show in the usage string
1114 1116 : std::string longest_name = m_names.front();
1115 1155 : for (const auto &s : m_names) {
1116 597 : if (s.size() > longest_name.size()) {
1117 39 : longest_name = s;
1118 : }
1119 : }
1120 558 : if (!m_is_required) {
1121 558 : usage << "[";
1122 : }
1123 558 : usage << longest_name;
1124 1116 : const std::string metavar = !m_metavar.empty() ? m_metavar : "VAR";
1125 558 : if (m_num_args_range.get_max() > 0) {
1126 352 : usage << " " << metavar;
1127 389 : if (m_num_args_range.get_max() > 1 &&
1128 37 : m_metavar.find("> <") == std::string::npos) {
1129 0 : usage << "...";
1130 : }
1131 : }
1132 558 : if (!m_is_required) {
1133 558 : usage << "]";
1134 : }
1135 558 : if (m_is_repeatable) {
1136 84 : usage << "...";
1137 : }
1138 1116 : return usage.str();
1139 : }
1140 :
1141 0 : std::size_t get_arguments_length() const {
1142 :
1143 0 : std::size_t names_size = std::accumulate(
1144 0 : std::begin(m_names), std::end(m_names), std::size_t(0),
1145 0 : [](const auto &sum, const auto &s) { return sum + s.size(); });
1146 :
1147 0 : if (is_positional(m_names.front(), m_prefix_chars)) {
1148 : // A set metavar means this replaces the names
1149 0 : if (!m_metavar.empty()) {
1150 : // Indent and metavar
1151 0 : return 2 + m_metavar.size();
1152 : }
1153 :
1154 : // Indent and space-separated
1155 0 : return 2 + names_size + (m_names.size() - 1);
1156 : }
1157 : // Is an option - include both names _and_ metavar
1158 : // size = text + (", " between names)
1159 0 : std::size_t size = names_size + 2 * (m_names.size() - 1);
1160 0 : if (!m_metavar.empty() && m_num_args_range == NArgsRange{1, 1}) {
1161 0 : size += m_metavar.size() + 1;
1162 : }
1163 0 : return size + 2; // indent
1164 : }
1165 :
1166 0 : friend std::ostream &operator<<(std::ostream &stream,
1167 : const Argument &argument) {
1168 0 : std::stringstream name_stream;
1169 0 : name_stream << " "; // indent
1170 0 : if (argument.is_positional(argument.m_names.front(),
1171 : argument.m_prefix_chars)) {
1172 0 : if (!argument.m_metavar.empty()) {
1173 0 : name_stream << argument.m_metavar;
1174 : } else {
1175 0 : name_stream << details::join(argument.m_names.begin(),
1176 0 : argument.m_names.end(), " ");
1177 : }
1178 : } else {
1179 0 : name_stream << details::join(argument.m_names.begin(),
1180 0 : argument.m_names.end(), ", ");
1181 : // If we have a metavar, and one narg - print the metavar
1182 0 : if (!argument.m_metavar.empty() &&
1183 0 : argument.m_num_args_range == NArgsRange{1, 1}) {
1184 0 : name_stream << " " << argument.m_metavar;
1185 : }
1186 0 : else if (!argument.m_metavar.empty() &&
1187 0 : argument.m_num_args_range.get_min() == argument.m_num_args_range.get_max() &&
1188 0 : argument.m_metavar.find("> <") != std::string::npos) {
1189 0 : name_stream << " " << argument.m_metavar;
1190 : }
1191 : }
1192 :
1193 : // align multiline help message
1194 0 : auto stream_width = stream.width();
1195 0 : auto name_padding = std::string(name_stream.str().size(), ' ');
1196 0 : auto pos = std::string::size_type{};
1197 0 : auto prev = std::string::size_type{};
1198 0 : auto first_line = true;
1199 0 : auto hspace = " "; // minimal space between name and help message
1200 0 : stream << name_stream.str();
1201 0 : std::string_view help_view(argument.m_help);
1202 0 : while ((pos = argument.m_help.find('\n', prev)) != std::string::npos) {
1203 0 : auto line = help_view.substr(prev, pos - prev + 1);
1204 0 : if (first_line) {
1205 0 : stream << hspace << line;
1206 0 : first_line = false;
1207 : } else {
1208 0 : stream.width(stream_width);
1209 0 : stream << name_padding << hspace << line;
1210 : }
1211 0 : prev += pos - prev + 1;
1212 : }
1213 0 : if (first_line) {
1214 0 : stream << hspace << argument.m_help;
1215 : } else {
1216 0 : auto leftover = help_view.substr(prev, argument.m_help.size() - prev);
1217 0 : if (!leftover.empty()) {
1218 0 : stream.width(stream_width);
1219 0 : stream << name_padding << hspace << leftover;
1220 : }
1221 : }
1222 :
1223 : // print nargs spec
1224 0 : if (!argument.m_help.empty()) {
1225 0 : stream << " ";
1226 : }
1227 0 : stream << argument.m_num_args_range;
1228 :
1229 0 : bool add_space = false;
1230 0 : if (argument.m_default_value.has_value() &&
1231 0 : argument.m_num_args_range != NArgsRange{0, 0}) {
1232 0 : stream << "[default: " << argument.m_default_value_repr << "]";
1233 0 : add_space = true;
1234 0 : } else if (argument.m_is_required) {
1235 0 : stream << "[required]";
1236 0 : add_space = true;
1237 : }
1238 0 : if (argument.m_is_repeatable) {
1239 0 : if (add_space) {
1240 0 : stream << " ";
1241 : }
1242 0 : stream << "[may be repeated]";
1243 : }
1244 0 : stream << "\n";
1245 0 : return stream;
1246 : }
1247 :
1248 : template <typename T> bool operator!=(const T &rhs) const {
1249 : return !(*this == rhs);
1250 : }
1251 :
1252 : /*
1253 : * Compare to an argument value of known type
1254 : * @throws std::logic_error in case of incompatible types
1255 : */
1256 : template <typename T> bool operator==(const T &rhs) const {
1257 : if constexpr (!details::IsContainer<T>) {
1258 : return get<T>() == rhs;
1259 : } else {
1260 : using ValueType = typename T::value_type;
1261 : auto lhs = get<T>();
1262 : return std::equal(std::begin(lhs), std::end(lhs), std::begin(rhs),
1263 : std::end(rhs), [](const auto &a, const auto &b) {
1264 : return std::any_cast<const ValueType &>(a) == b;
1265 : });
1266 : }
1267 : }
1268 :
1269 : /*
1270 : * positional:
1271 : * _empty_
1272 : * '-'
1273 : * '-' decimal-literal
1274 : * !'-' anything
1275 : */
1276 300707 : static bool is_positional(std::string_view name,
1277 : std::string_view prefix_chars) {
1278 300707 : auto first = lookahead(name);
1279 :
1280 300707 : if (first == eof) {
1281 3 : return true;
1282 : }
1283 300704 : if (prefix_chars.find(static_cast<char>(first)) !=
1284 : std::string_view::npos) {
1285 263112 : name.remove_prefix(1);
1286 263112 : if (name.empty()) {
1287 0 : return true;
1288 : }
1289 263112 : return is_decimal_literal(name);
1290 : }
1291 37592 : return true;
1292 : }
1293 :
1294 : private:
1295 : class NArgsRange {
1296 : std::size_t m_min;
1297 : std::size_t m_max;
1298 :
1299 : public:
1300 460143 : NArgsRange(std::size_t minimum, std::size_t maximum)
1301 460143 : : m_min(minimum), m_max(maximum) {
1302 460143 : if (minimum > maximum) {
1303 0 : throw std::logic_error("Range of number of arguments is invalid");
1304 : }
1305 460143 : }
1306 :
1307 2359 : bool contains(std::size_t value) const {
1308 2359 : return value >= m_min && value <= m_max;
1309 : }
1310 :
1311 5 : bool is_exact() const { return m_min == m_max; }
1312 :
1313 29 : bool is_right_bounded() const {
1314 29 : return m_max < (std::numeric_limits<std::size_t>::max)();
1315 : }
1316 :
1317 25975 : std::size_t get_min() const { return m_min; }
1318 :
1319 123857 : std::size_t get_max() const { return m_max; }
1320 :
1321 : // Print help message
1322 0 : friend auto operator<<(std::ostream &stream, const NArgsRange &range)
1323 : -> std::ostream & {
1324 0 : if (range.m_min == range.m_max) {
1325 0 : if (range.m_min != 0 && range.m_min != 1) {
1326 0 : stream << "[nargs: " << range.m_min << "] ";
1327 : }
1328 : } else {
1329 0 : if (range.m_max == (std::numeric_limits<std::size_t>::max)()) {
1330 0 : stream << "[nargs: " << range.m_min << " or more] ";
1331 : } else {
1332 0 : stream << "[nargs=" << range.m_min << ".." << range.m_max << "] ";
1333 : }
1334 : }
1335 0 : return stream;
1336 : }
1337 :
1338 0 : bool operator==(const NArgsRange &rhs) const {
1339 0 : return rhs.m_min == m_min && rhs.m_max == m_max;
1340 : }
1341 :
1342 0 : bool operator!=(const NArgsRange &rhs) const { return !(*this == rhs); }
1343 : };
1344 :
1345 5 : void throw_nargs_range_validation_error() const {
1346 10 : std::stringstream stream;
1347 5 : if (!m_used_name.empty()) {
1348 0 : stream << m_used_name << ": ";
1349 : } else {
1350 5 : stream << m_names.front() << ": ";
1351 : }
1352 5 : if (m_num_args_range.is_exact()) {
1353 5 : stream << m_num_args_range.get_min();
1354 0 : } else if (m_num_args_range.is_right_bounded()) {
1355 0 : stream << m_num_args_range.get_min() << " to "
1356 0 : << m_num_args_range.get_max();
1357 : } else {
1358 0 : stream << m_num_args_range.get_min() << " or more";
1359 : }
1360 5 : stream << " argument(s) expected. " << m_values.size() << " provided.";
1361 5 : throw std::runtime_error(stream.str());
1362 : }
1363 :
1364 0 : void throw_required_arg_not_used_error() const {
1365 0 : std::stringstream stream;
1366 0 : stream << m_names.front() << ": required.";
1367 0 : throw std::runtime_error(stream.str());
1368 : }
1369 :
1370 0 : void throw_required_arg_no_value_provided_error() const {
1371 0 : std::stringstream stream;
1372 0 : stream << m_used_name << ": no value provided.";
1373 0 : throw std::runtime_error(stream.str());
1374 : }
1375 :
1376 : static constexpr int eof = std::char_traits<char>::eof();
1377 :
1378 566620 : static auto lookahead(std::string_view s) -> int {
1379 566620 : if (s.empty()) {
1380 679 : return eof;
1381 : }
1382 565941 : return static_cast<int>(static_cast<unsigned char>(s[0]));
1383 : }
1384 :
1385 : /*
1386 : * decimal-literal:
1387 : * '0'
1388 : * nonzero-digit digit-sequence_opt
1389 : * integer-part fractional-part
1390 : * fractional-part
1391 : * integer-part '.' exponent-part_opt
1392 : * integer-part exponent-part
1393 : *
1394 : * integer-part:
1395 : * digit-sequence
1396 : *
1397 : * fractional-part:
1398 : * '.' post-decimal-point
1399 : *
1400 : * post-decimal-point:
1401 : * digit-sequence exponent-part_opt
1402 : *
1403 : * exponent-part:
1404 : * 'e' post-e
1405 : * 'E' post-e
1406 : *
1407 : * post-e:
1408 : * sign_opt digit-sequence
1409 : *
1410 : * sign: one of
1411 : * '+' '-'
1412 : */
1413 263112 : static bool is_decimal_literal(std::string_view s) {
1414 263112 : if (s == "inf") {
1415 2 : return true;
1416 : }
1417 :
1418 13224 : auto is_digit = [](auto c) constexpr {
1419 13224 : switch (c) {
1420 12473 : case '0':
1421 : case '1':
1422 : case '2':
1423 : case '3':
1424 : case '4':
1425 : case '5':
1426 : case '6':
1427 : case '7':
1428 : case '8':
1429 : case '9':
1430 12473 : return true;
1431 751 : default:
1432 751 : return false;
1433 : }
1434 : };
1435 :
1436 : // precondition: we have consumed or will consume at least one digit
1437 1643 : auto consume_digits = [=](std::string_view sd) {
1438 : // NOLINTNEXTLINE(readability-qualified-auto)
1439 1643 : auto it = std::find_if_not(std::begin(sd), std::end(sd), is_digit);
1440 1643 : return sd.substr(static_cast<std::size_t>(it - std::begin(sd)));
1441 : };
1442 :
1443 263110 : switch (lookahead(s)) {
1444 48 : case '0': {
1445 48 : s.remove_prefix(1);
1446 48 : if (s.empty()) {
1447 0 : return true;
1448 : }
1449 48 : goto integer_part;
1450 : }
1451 907 : case '1':
1452 : case '2':
1453 : case '3':
1454 : case '4':
1455 : case '5':
1456 : case '6':
1457 : case '7':
1458 : case '8':
1459 : case '9': {
1460 907 : s = consume_digits(s);
1461 907 : if (s.empty()) {
1462 210 : return true;
1463 : }
1464 697 : goto integer_part_consumed;
1465 : }
1466 0 : case '.': {
1467 0 : s.remove_prefix(1);
1468 0 : goto post_decimal_point;
1469 : }
1470 262155 : default:
1471 262155 : return false;
1472 : }
1473 :
1474 48 : integer_part:
1475 48 : s = consume_digits(s);
1476 745 : integer_part_consumed:
1477 745 : switch (lookahead(s)) {
1478 682 : case '.': {
1479 682 : s.remove_prefix(1);
1480 682 : if (is_digit(lookahead(s))) {
1481 682 : goto post_decimal_point;
1482 : } else {
1483 0 : goto exponent_part_opt;
1484 : }
1485 : }
1486 0 : case 'e':
1487 : case 'E': {
1488 0 : s.remove_prefix(1);
1489 0 : goto post_e;
1490 : }
1491 63 : default:
1492 63 : return false;
1493 : }
1494 :
1495 682 : post_decimal_point:
1496 682 : if (is_digit(lookahead(s))) {
1497 682 : s = consume_digits(s);
1498 682 : goto exponent_part_opt;
1499 : }
1500 0 : return false;
1501 :
1502 682 : exponent_part_opt:
1503 682 : switch (lookahead(s)) {
1504 676 : case eof:
1505 676 : return true;
1506 6 : case 'e':
1507 : case 'E': {
1508 6 : s.remove_prefix(1);
1509 6 : goto post_e;
1510 : }
1511 0 : default:
1512 0 : return false;
1513 : }
1514 :
1515 6 : post_e:
1516 6 : switch (lookahead(s)) {
1517 6 : case '-':
1518 : case '+':
1519 6 : s.remove_prefix(1);
1520 : }
1521 6 : if (is_digit(lookahead(s))) {
1522 6 : s = consume_digits(s);
1523 6 : return s.empty();
1524 : }
1525 0 : return false;
1526 : }
1527 :
1528 274593 : static bool is_optional(std::string_view name,
1529 : std::string_view prefix_chars) {
1530 274593 : return !is_positional(name, prefix_chars);
1531 : }
1532 :
1533 : /*
1534 : * Get argument value given a type
1535 : * @throws std::logic_error in case of incompatible types
1536 : */
1537 4 : template <typename T> T get() const {
1538 4 : if (!m_values.empty()) {
1539 : if constexpr (details::IsContainer<T>) {
1540 : return any_cast_container<T>(m_values);
1541 : } else {
1542 4 : return std::any_cast<T>(m_values.front());
1543 : }
1544 : }
1545 0 : if (m_default_value.has_value()) {
1546 0 : return std::any_cast<T>(m_default_value);
1547 : }
1548 : if constexpr (details::IsContainer<T>) {
1549 : if (!m_accepts_optional_like_value) {
1550 : return any_cast_container<T>(m_values);
1551 : }
1552 : }
1553 :
1554 0 : throw std::logic_error("No value provided for '" + m_names.back() + "'.");
1555 : }
1556 :
1557 : /*
1558 : * Get argument value given a type.
1559 : * @pre The object has no default value.
1560 : * @returns The stored value if any, std::nullopt otherwise.
1561 : */
1562 19999 : template <typename T> auto present() const -> std::optional<T> {
1563 19999 : if (m_default_value.has_value()) {
1564 0 : throw std::logic_error("Argument with default value always presents");
1565 : }
1566 19999 : if (m_values.empty()) {
1567 18035 : return std::nullopt;
1568 : }
1569 : if constexpr (details::IsContainer<T>) {
1570 1964 : return any_cast_container<T>(m_values);
1571 : }
1572 : return std::any_cast<T>(m_values.front());
1573 : }
1574 :
1575 : template <typename T>
1576 1964 : static auto any_cast_container(const std::vector<std::any> &operand) -> T {
1577 : using ValueType = typename T::value_type;
1578 :
1579 1964 : T result;
1580 1964 : std::transform(
1581 : std::begin(operand), std::end(operand), std::back_inserter(result),
1582 6317 : [](const auto &value) { return std::any_cast<ValueType>(value); });
1583 1964 : return result;
1584 : }
1585 :
1586 279888 : void set_usage_newline_counter(int i) { m_usage_newline_counter = i; }
1587 :
1588 279888 : void set_group_idx(std::size_t i) { m_group_idx = i; }
1589 :
1590 : std::vector<std::string> m_names;
1591 : std::string_view m_used_name;
1592 : std::string m_help;
1593 : std::string m_metavar;
1594 : std::any m_default_value;
1595 : std::string m_default_value_repr;
1596 : std::optional<std::string>
1597 : m_default_value_str; // used for checking default_value against choices
1598 : std::any m_implicit_value;
1599 : std::optional<std::vector<std::string>> m_choices{std::nullopt};
1600 : using valued_action = std::function<std::any(const std::string &)>;
1601 : using void_action = std::function<void(const std::string &)>;
1602 : std::vector<std::variant<valued_action, void_action>> m_actions;
1603 : std::variant<valued_action, void_action> m_default_action{
1604 : std::in_place_type<valued_action>,
1605 512 : [](const std::string &value) { return value; }};
1606 : std::vector<std::any> m_values;
1607 : NArgsRange m_num_args_range{1, 1};
1608 : // Bit field of bool values. Set default value in ctor.
1609 : bool m_accepts_optional_like_value : 1;
1610 : bool m_is_optional : 1;
1611 : bool m_is_required : 1;
1612 : bool m_is_repeatable : 1;
1613 : bool m_is_used : 1;
1614 : bool m_is_hidden : 1; // if set, does not appear in usage or help
1615 : std::string_view m_prefix_chars; // ArgumentParser has the prefix_chars
1616 : int m_usage_newline_counter = 0;
1617 : std::size_t m_group_idx = 0;
1618 : };
1619 :
1620 : class ArgumentParser {
1621 : public:
1622 5601 : explicit ArgumentParser(std::string program_name = {},
1623 : std::string version = "1.0",
1624 : default_arguments add_args = default_arguments::all,
1625 : bool exit_on_default_arguments = true,
1626 : std::ostream &os = std::cout)
1627 11202 : : m_program_name(std::move(program_name)), m_version(std::move(version)),
1628 : m_exit_on_default_arguments(exit_on_default_arguments),
1629 5601 : m_parser_path(m_program_name) {
1630 5601 : if ((add_args & default_arguments::help) == default_arguments::help) {
1631 0 : add_argument("-h", "--help")
1632 0 : .action([&](const auto & /*unused*/) {
1633 0 : os << help().str();
1634 0 : if (m_exit_on_default_arguments) {
1635 0 : std::exit(0);
1636 : }
1637 0 : })
1638 0 : .default_value(false)
1639 0 : .help("shows help message and exits")
1640 0 : .implicit_value(true)
1641 0 : .nargs(0);
1642 : }
1643 5601 : if ((add_args & default_arguments::version) == default_arguments::version) {
1644 0 : add_argument("-v", "--version")
1645 0 : .action([&](const auto & /*unused*/) {
1646 0 : os << m_version << std::endl;
1647 0 : if (m_exit_on_default_arguments) {
1648 0 : std::exit(0);
1649 : }
1650 0 : })
1651 0 : .default_value(false)
1652 0 : .help("prints version information and exits")
1653 0 : .implicit_value(true)
1654 0 : .nargs(0);
1655 : }
1656 5601 : }
1657 :
1658 5486 : ~ArgumentParser() = default;
1659 :
1660 : // ArgumentParser is meant to be used in a single function.
1661 : // Setup everything and parse arguments in one place.
1662 : //
1663 : // ArgumentParser internally uses std::string_views,
1664 : // references, iterators, etc.
1665 : // Many of these elements become invalidated after a copy or move.
1666 : ArgumentParser(const ArgumentParser &other) = delete;
1667 : ArgumentParser &operator=(const ArgumentParser &other) = delete;
1668 : ArgumentParser(ArgumentParser &&) noexcept = delete;
1669 : ArgumentParser &operator=(ArgumentParser &&) = delete;
1670 :
1671 : explicit operator bool() const {
1672 : auto arg_used = std::any_of(m_argument_map.cbegin(), m_argument_map.cend(),
1673 : [](auto &it) { return it.second->m_is_used; });
1674 : auto subparser_used =
1675 : std::any_of(m_subparser_used.cbegin(), m_subparser_used.cend(),
1676 : [](auto &it) { return it.second; });
1677 :
1678 : return m_is_parsed && (arg_used || subparser_used);
1679 : }
1680 :
1681 : // Parameter packing
1682 : // Call add_argument with variadic number of string arguments
1683 243420 : template <typename... Targs> Argument &add_argument(Targs... f_args) {
1684 : using array_of_sv = std::array<std::string_view, sizeof...(Targs)>;
1685 486840 : auto argument =
1686 243420 : m_optional_arguments.emplace(std::cend(m_optional_arguments),
1687 243420 : m_prefix_chars, array_of_sv{f_args...});
1688 :
1689 243420 : if (!argument->m_is_optional) {
1690 5584 : m_positional_arguments.splice(std::cend(m_positional_arguments),
1691 2792 : m_optional_arguments, argument);
1692 : }
1693 243420 : argument->set_usage_newline_counter(m_usage_newline_counter);
1694 243420 : argument->set_group_idx(m_group_names.size());
1695 :
1696 243420 : index_argument(argument);
1697 486840 : return *argument;
1698 : }
1699 :
1700 : class MutuallyExclusiveGroup {
1701 : friend class ArgumentParser;
1702 :
1703 : public:
1704 : MutuallyExclusiveGroup() = delete;
1705 :
1706 16143 : explicit MutuallyExclusiveGroup(ArgumentParser &parent,
1707 : bool required = false)
1708 16143 : : m_parent(parent), m_required(required), m_elements({}) {}
1709 :
1710 : MutuallyExclusiveGroup(const MutuallyExclusiveGroup &other) = delete;
1711 : MutuallyExclusiveGroup &
1712 : operator=(const MutuallyExclusiveGroup &other) = delete;
1713 :
1714 12200 : MutuallyExclusiveGroup(MutuallyExclusiveGroup &&other) noexcept
1715 12200 : : m_parent(other.m_parent), m_required(other.m_required),
1716 12200 : m_elements(std::move(other.m_elements)) {
1717 12200 : other.m_elements.clear();
1718 12200 : }
1719 :
1720 36468 : template <typename... Targs> Argument &add_argument(Targs... f_args) {
1721 36468 : auto &argument = m_parent.add_argument(std::forward<Targs>(f_args)...);
1722 36468 : m_elements.push_back(&argument);
1723 36468 : argument.set_usage_newline_counter(m_parent.m_usage_newline_counter);
1724 36468 : argument.set_group_idx(m_parent.m_group_names.size());
1725 36468 : return argument;
1726 : }
1727 :
1728 : private:
1729 : ArgumentParser &m_parent;
1730 : bool m_required{false};
1731 : std::vector<Argument *> m_elements{};
1732 : };
1733 :
1734 16143 : MutuallyExclusiveGroup &add_mutually_exclusive_group(bool required = false) {
1735 16143 : m_mutually_exclusive_groups.emplace_back(*this, required);
1736 16143 : return m_mutually_exclusive_groups.back();
1737 : }
1738 :
1739 : // Parameter packed add_parents method
1740 : // Accepts a variadic number of ArgumentParser objects
1741 : template <typename... Targs>
1742 : ArgumentParser &add_parents(const Targs &... f_args) {
1743 : for (const ArgumentParser &parent_parser : {std::ref(f_args)...}) {
1744 : for (const auto &argument : parent_parser.m_positional_arguments) {
1745 : auto it = m_positional_arguments.insert(
1746 : std::cend(m_positional_arguments), argument);
1747 : index_argument(it);
1748 : }
1749 : for (const auto &argument : parent_parser.m_optional_arguments) {
1750 : auto it = m_optional_arguments.insert(std::cend(m_optional_arguments),
1751 : argument);
1752 : index_argument(it);
1753 : }
1754 : }
1755 : return *this;
1756 : }
1757 :
1758 : // Ask for the next optional arguments to be displayed on a separate
1759 : // line in usage() output. Only effective if set_usage_max_line_width() is
1760 : // also used.
1761 8483 : ArgumentParser &add_usage_newline() {
1762 8483 : ++m_usage_newline_counter;
1763 8483 : return *this;
1764 : }
1765 :
1766 : // Ask for the next optional arguments to be displayed in a separate section
1767 : // in usage() and help (<< *this) output.
1768 : // For usage(), this is only effective if set_usage_max_line_width() is
1769 : // also used.
1770 3314 : ArgumentParser &add_group(std::string group_name) {
1771 3314 : m_group_names.emplace_back(std::move(group_name));
1772 3314 : return *this;
1773 : }
1774 :
1775 5601 : ArgumentParser &add_description(std::string description) {
1776 5601 : m_description = std::move(description);
1777 5601 : return *this;
1778 : }
1779 :
1780 5006 : ArgumentParser &add_epilog(std::string epilog) {
1781 5006 : m_epilog = std::move(epilog);
1782 5006 : return *this;
1783 : }
1784 :
1785 : // Add a un-documented/hidden alias for an argument.
1786 : // Ideally we'd want this to be a method of Argument, but Argument
1787 : // does not own its owing ArgumentParser.
1788 7151 : ArgumentParser &add_hidden_alias_for(Argument &arg, std::string_view alias) {
1789 84191 : for (auto it = m_optional_arguments.begin();
1790 161231 : it != m_optional_arguments.end(); ++it) {
1791 84191 : if (&(*it) == &arg) {
1792 7151 : m_argument_map.insert_or_assign(std::string(alias), it);
1793 7151 : return *this;
1794 : }
1795 : }
1796 : throw std::logic_error(
1797 0 : "Argument is not an optional argument of this parser");
1798 : }
1799 :
1800 : /* Getter for arguments and subparsers.
1801 : * @throws std::logic_error in case of an invalid argument or subparser name
1802 : */
1803 : template <typename T = Argument> T &at(std::string_view name) {
1804 : if constexpr (std::is_same_v<T, Argument>) {
1805 : return (*this)[name];
1806 : } else {
1807 : std::string str_name(name);
1808 : auto subparser_it = m_subparser_map.find(str_name);
1809 : if (subparser_it != m_subparser_map.end()) {
1810 : return subparser_it->second->get();
1811 : }
1812 : throw std::logic_error("No such subparser: " + str_name);
1813 : }
1814 : }
1815 :
1816 : ArgumentParser &set_prefix_chars(std::string prefix_chars) {
1817 : m_prefix_chars = std::move(prefix_chars);
1818 : return *this;
1819 : }
1820 :
1821 : ArgumentParser &set_assign_chars(std::string assign_chars) {
1822 : m_assign_chars = std::move(assign_chars);
1823 : return *this;
1824 : }
1825 :
1826 : /* Call parse_args_internal - which does all the work
1827 : * Then, validate the parsed arguments
1828 : * This variant is used mainly for testing
1829 : * @throws std::runtime_error in case of any invalid argument
1830 : */
1831 4991 : void parse_args(const std::vector<std::string> &arguments) {
1832 4991 : parse_args_internal(arguments);
1833 : // Check if all arguments are parsed
1834 255244 : for ([[maybe_unused]] const auto &[unused, argument] : m_argument_map) {
1835 250289 : argument->validate();
1836 : }
1837 :
1838 : // Check each mutually exclusive group and make sure
1839 : // there are no constraint violations
1840 20933 : for (const auto &group : m_mutually_exclusive_groups) {
1841 15979 : auto mutex_argument_used{false};
1842 15979 : Argument *mutex_argument_it{nullptr};
1843 52052 : for (Argument *arg : group.m_elements) {
1844 36074 : if (!mutex_argument_used && arg->m_is_used) {
1845 545 : mutex_argument_used = true;
1846 545 : mutex_argument_it = arg;
1847 35529 : } else if (mutex_argument_used && arg->m_is_used) {
1848 : // Violation
1849 2 : throw std::runtime_error("Argument '" + arg->get_usage_full() +
1850 2 : "' not allowed with '" +
1851 3 : mutex_argument_it->get_usage_full() + "'");
1852 : }
1853 : }
1854 :
1855 15978 : if (!mutex_argument_used && group.m_required) {
1856 : // at least one argument from the group is
1857 : // required
1858 0 : std::string argument_names{};
1859 0 : std::size_t i = 0;
1860 0 : std::size_t size = group.m_elements.size();
1861 0 : for (Argument *arg : group.m_elements) {
1862 0 : if (i + 1 == size) {
1863 : // last
1864 0 : argument_names += "'" + arg->get_usage_full() + "' ";
1865 : } else {
1866 0 : argument_names += "'" + arg->get_usage_full() + "' or ";
1867 : }
1868 0 : i += 1;
1869 : }
1870 0 : throw std::runtime_error("One of the arguments " + argument_names +
1871 0 : "is required");
1872 : }
1873 : }
1874 4954 : }
1875 :
1876 : /* Call parse_known_args_internal - which does all the work
1877 : * Then, validate the parsed arguments
1878 : * This variant is used mainly for testing
1879 : * @throws std::runtime_error in case of any invalid argument
1880 : */
1881 : std::vector<std::string>
1882 : parse_known_args(const std::vector<std::string> &arguments) {
1883 : auto unknown_arguments = parse_known_args_internal(arguments);
1884 : // Check if all arguments are parsed
1885 : for ([[maybe_unused]] const auto &[unused, argument] : m_argument_map) {
1886 : argument->validate();
1887 : }
1888 : return unknown_arguments;
1889 : }
1890 :
1891 : /* Main entry point for parsing command-line arguments using this
1892 : * ArgumentParser
1893 : * @throws std::runtime_error in case of any invalid argument
1894 : */
1895 : // NOLINTNEXTLINE(cppcoreguidelines-avoid-c-arrays)
1896 : void parse_args(int argc, const char *const argv[]) {
1897 : parse_args({argv, argv + argc});
1898 : }
1899 :
1900 : /* Main entry point for parsing command-line arguments using this
1901 : * ArgumentParser
1902 : * @throws std::runtime_error in case of any invalid argument
1903 : */
1904 : // NOLINTNEXTLINE(cppcoreguidelines-avoid-c-arrays)
1905 : auto parse_known_args(int argc, const char *const argv[]) {
1906 : return parse_known_args({argv, argv + argc});
1907 : }
1908 :
1909 : /* Getter for options with default values.
1910 : * @throws std::logic_error if parse_args() has not been previously called
1911 : * @throws std::logic_error if there is no such option
1912 : * @throws std::logic_error if the option has no value
1913 : * @throws std::bad_any_cast if the option is not of type T
1914 : */
1915 4 : template <typename T = std::string> T get(std::string_view arg_name) const {
1916 4 : if (!m_is_parsed) {
1917 0 : throw std::logic_error("Nothing parsed, no arguments are available.");
1918 : }
1919 4 : return (*this)[arg_name].get<T>();
1920 : }
1921 :
1922 : /* Getter for options without default values.
1923 : * @pre The option has no default value.
1924 : * @throws std::logic_error if there is no such option
1925 : * @throws std::bad_any_cast if the option is not of type T
1926 : */
1927 : template <typename T = std::string>
1928 19999 : auto present(std::string_view arg_name) const -> std::optional<T> {
1929 19999 : return (*this)[arg_name].present<T>();
1930 : }
1931 :
1932 : /* Getter that returns true for user-supplied options. Returns false if not
1933 : * user-supplied, even with a default value.
1934 : */
1935 4793 : auto is_used(std::string_view arg_name) const {
1936 4793 : return (*this)[arg_name].m_is_used;
1937 : }
1938 :
1939 : /* Getter that returns true if a subcommand is used.
1940 : */
1941 30 : auto is_subcommand_used(std::string_view subcommand_name) const {
1942 30 : return m_subparser_used.at(std::string(subcommand_name));
1943 : }
1944 :
1945 : /* Getter that returns true if a subcommand is used.
1946 : */
1947 : auto is_subcommand_used(const ArgumentParser &subparser) const {
1948 : return is_subcommand_used(subparser.m_program_name);
1949 : }
1950 :
1951 : /* Indexing operator. Return a reference to an Argument object
1952 : * Used in conjunction with Argument.operator== e.g., parser["foo"] == true
1953 : * @throws std::logic_error in case of an invalid argument name
1954 : */
1955 24796 : Argument &operator[](std::string_view arg_name) const {
1956 49592 : std::string name(arg_name);
1957 24796 : auto it = m_argument_map.find(name);
1958 24796 : if (it != m_argument_map.end()) {
1959 24776 : return *(it->second);
1960 : }
1961 20 : if (!is_valid_prefix_char(arg_name.front())) {
1962 0 : const auto legal_prefix_char = get_any_valid_prefix_char();
1963 0 : const auto prefix = std::string(1, legal_prefix_char);
1964 :
1965 : // "-" + arg_name
1966 0 : name = prefix + name;
1967 0 : it = m_argument_map.find(name);
1968 0 : if (it != m_argument_map.end()) {
1969 0 : return *(it->second);
1970 : }
1971 : // "--" + arg_name
1972 0 : name = prefix + name;
1973 0 : it = m_argument_map.find(name);
1974 0 : if (it != m_argument_map.end()) {
1975 0 : return *(it->second);
1976 : }
1977 : }
1978 20 : throw std::logic_error("No such argument: " + std::string(arg_name));
1979 : }
1980 :
1981 : // Print help message
1982 0 : friend auto operator<<(std::ostream &stream, const ArgumentParser &parser)
1983 : -> std::ostream & {
1984 0 : stream.setf(std::ios_base::left);
1985 :
1986 0 : auto longest_arg_length = parser.get_length_of_longest_argument();
1987 :
1988 0 : stream << parser.usage() << "\n\n";
1989 :
1990 0 : if (!parser.m_description.empty()) {
1991 0 : stream << parser.m_description << "\n\n";
1992 : }
1993 :
1994 0 : const bool has_visible_positional_args = std::find_if(
1995 : parser.m_positional_arguments.begin(),
1996 : parser.m_positional_arguments.end(),
1997 0 : [](const auto &argument) {
1998 0 : return !argument.m_is_hidden; }) !=
1999 0 : parser.m_positional_arguments.end();
2000 0 : if (has_visible_positional_args) {
2001 0 : stream << "Positional arguments:\n";
2002 : }
2003 :
2004 0 : for (const auto &argument : parser.m_positional_arguments) {
2005 0 : if (!argument.m_is_hidden) {
2006 0 : stream.width(static_cast<std::streamsize>(longest_arg_length));
2007 0 : stream << argument;
2008 : }
2009 : }
2010 :
2011 0 : if (!parser.m_optional_arguments.empty()) {
2012 : stream << (!has_visible_positional_args ? "" : "\n")
2013 0 : << "Optional arguments:\n";
2014 : }
2015 :
2016 0 : for (const auto &argument : parser.m_optional_arguments) {
2017 0 : if (argument.m_group_idx == 0 && !argument.m_is_hidden) {
2018 0 : stream.width(static_cast<std::streamsize>(longest_arg_length));
2019 0 : stream << argument;
2020 : }
2021 : }
2022 :
2023 0 : for (size_t i_group = 0; i_group < parser.m_group_names.size(); ++i_group) {
2024 0 : stream << "\n" << parser.m_group_names[i_group] << " (detailed usage):\n";
2025 0 : for (const auto &argument : parser.m_optional_arguments) {
2026 0 : if (argument.m_group_idx == i_group + 1 && !argument.m_is_hidden) {
2027 0 : stream.width(static_cast<std::streamsize>(longest_arg_length));
2028 0 : stream << argument;
2029 : }
2030 : }
2031 : }
2032 :
2033 0 : bool has_visible_subcommands = std::any_of(
2034 : parser.m_subparser_map.begin(), parser.m_subparser_map.end(),
2035 0 : [](auto &p) { return !p.second->get().m_suppress; });
2036 :
2037 0 : if (has_visible_subcommands) {
2038 0 : stream << (parser.m_positional_arguments.empty()
2039 0 : ? (parser.m_optional_arguments.empty() ? "" : "\n")
2040 : : "\n")
2041 0 : << "Subcommands:\n";
2042 0 : for (const auto &[command, subparser] : parser.m_subparser_map) {
2043 0 : if (subparser->get().m_suppress) {
2044 0 : continue;
2045 : }
2046 :
2047 0 : stream << std::setw(2) << " ";
2048 0 : if (longest_arg_length >= 2) {
2049 0 : stream << std::setw(static_cast<int>(longest_arg_length - 2))
2050 0 : << command;
2051 : }
2052 0 : stream << " " << subparser->get().m_description << "\n";
2053 : }
2054 : }
2055 :
2056 0 : if (!parser.m_epilog.empty()) {
2057 0 : stream << '\n';
2058 0 : stream << parser.m_epilog << "\n\n";
2059 : }
2060 :
2061 0 : return stream;
2062 : }
2063 :
2064 : // Format help message
2065 0 : auto help() const -> std::stringstream {
2066 0 : std::stringstream out;
2067 0 : out << *this;
2068 0 : return out;
2069 : }
2070 :
2071 : // Sets the maximum width for a line of the Usage message
2072 5601 : ArgumentParser &set_usage_max_line_width(size_t w) {
2073 5601 : this->m_usage_max_line_width = w;
2074 5601 : return *this;
2075 : }
2076 :
2077 : // Asks to display arguments of mutually exclusive group on separate lines in
2078 : // the Usage message
2079 5601 : ArgumentParser &set_usage_break_on_mutex() {
2080 5601 : this->m_usage_break_on_mutex = true;
2081 5601 : return *this;
2082 : }
2083 :
2084 : // Format usage part of help only
2085 20 : auto usage() const -> std::string {
2086 40 : std::stringstream stream;
2087 :
2088 40 : std::string curline("Usage: ");
2089 20 : curline += this->m_parser_path;
2090 : const bool multiline_usage =
2091 20 : this->m_usage_max_line_width < std::numeric_limits<std::size_t>::max();
2092 20 : const size_t indent_size = curline.size();
2093 :
2094 25 : const auto deal_with_options_of_group = [&](std::size_t group_idx) {
2095 25 : bool found_options = false;
2096 : // Add any options inline here
2097 25 : const MutuallyExclusiveGroup *cur_mutex = nullptr;
2098 25 : int usage_newline_counter = -1;
2099 1002 : for (const auto &argument : this->m_optional_arguments) {
2100 977 : if (argument.m_is_hidden) {
2101 419 : continue;
2102 : }
2103 898 : if (multiline_usage) {
2104 898 : if (argument.m_group_idx != group_idx) {
2105 340 : continue;
2106 : }
2107 558 : if (usage_newline_counter != argument.m_usage_newline_counter) {
2108 42 : if (usage_newline_counter >= 0) {
2109 17 : if (curline.size() > indent_size) {
2110 17 : stream << curline << std::endl;
2111 17 : curline = std::string(indent_size, ' ');
2112 : }
2113 : }
2114 42 : usage_newline_counter = argument.m_usage_newline_counter;
2115 : }
2116 : }
2117 558 : found_options = true;
2118 1116 : const std::string arg_inline_usage = argument.get_inline_usage();
2119 : const MutuallyExclusiveGroup *arg_mutex =
2120 558 : get_belonging_mutex(&argument);
2121 558 : if ((cur_mutex != nullptr) && (arg_mutex == nullptr)) {
2122 26 : curline += ']';
2123 26 : if (this->m_usage_break_on_mutex) {
2124 26 : stream << curline << std::endl;
2125 26 : curline = std::string(indent_size, ' ');
2126 : }
2127 532 : } else if ((cur_mutex == nullptr) && (arg_mutex != nullptr)) {
2128 28 : if ((this->m_usage_break_on_mutex && curline.size() > indent_size) ||
2129 1 : curline.size() + 3 + arg_inline_usage.size() >
2130 1 : this->m_usage_max_line_width) {
2131 26 : stream << curline << std::endl;
2132 26 : curline = std::string(indent_size, ' ');
2133 : }
2134 27 : curline += " [";
2135 505 : } else if ((cur_mutex != nullptr) && (arg_mutex != nullptr)) {
2136 42 : if (cur_mutex != arg_mutex) {
2137 4 : curline += ']';
2138 4 : if (this->m_usage_break_on_mutex ||
2139 0 : curline.size() + 3 + arg_inline_usage.size() >
2140 0 : this->m_usage_max_line_width) {
2141 4 : stream << curline << std::endl;
2142 4 : curline = std::string(indent_size, ' ');
2143 : }
2144 4 : curline += " [";
2145 : } else {
2146 38 : curline += '|';
2147 : }
2148 : }
2149 558 : cur_mutex = arg_mutex;
2150 558 : if (curline.size() + 1 + arg_inline_usage.size() >
2151 558 : this->m_usage_max_line_width) {
2152 97 : stream << curline << std::endl;
2153 97 : curline = std::string(indent_size, ' ');
2154 97 : curline += " ";
2155 461 : } else if (cur_mutex == nullptr) {
2156 396 : curline += " ";
2157 : }
2158 558 : curline += arg_inline_usage;
2159 : }
2160 25 : if (cur_mutex != nullptr) {
2161 1 : curline += ']';
2162 : }
2163 25 : return found_options;
2164 20 : };
2165 :
2166 20 : const bool found_options = deal_with_options_of_group(0);
2167 :
2168 40 : if (found_options && multiline_usage &&
2169 20 : !this->m_positional_arguments.empty()) {
2170 14 : stream << curline << std::endl;
2171 14 : curline = std::string(indent_size, ' ');
2172 : }
2173 : // Put positional arguments after the optionals
2174 49 : for (const auto &argument : this->m_positional_arguments) {
2175 29 : if (argument.m_is_hidden) {
2176 0 : continue;
2177 : }
2178 29 : const std::string pos_arg = !argument.m_metavar.empty()
2179 : ? argument.m_metavar
2180 58 : : argument.m_names.front();
2181 29 : if (curline.size() + 1 + pos_arg.size() > this->m_usage_max_line_width) {
2182 0 : stream << curline << std::endl;
2183 0 : curline = std::string(indent_size, ' ');
2184 : }
2185 29 : curline += " ";
2186 32 : if (argument.m_num_args_range.get_min() == 0 &&
2187 3 : !argument.m_num_args_range.is_right_bounded()) {
2188 3 : curline += "[";
2189 3 : curline += pos_arg;
2190 3 : curline += "]...";
2191 52 : } else if (argument.m_num_args_range.get_min() == 1 &&
2192 26 : !argument.m_num_args_range.is_right_bounded()) {
2193 4 : curline += pos_arg;
2194 4 : curline += "...";
2195 : } else {
2196 22 : curline += pos_arg;
2197 : }
2198 : }
2199 :
2200 20 : if (multiline_usage) {
2201 : // Display options of other groups
2202 25 : for (std::size_t i = 0; i < m_group_names.size(); ++i) {
2203 5 : stream << curline << std::endl << std::endl;
2204 5 : stream << m_group_names[i] << ":" << std::endl;
2205 5 : curline = std::string(indent_size, ' ');
2206 5 : deal_with_options_of_group(i + 1);
2207 : }
2208 : }
2209 :
2210 20 : stream << curline;
2211 :
2212 : // Put subcommands after positional arguments
2213 20 : if (!m_subparser_map.empty()) {
2214 5 : stream << " {";
2215 5 : std::size_t i{0};
2216 28 : for (const auto &[command, subparser] : m_subparser_map) {
2217 23 : if (subparser->get().m_suppress) {
2218 0 : continue;
2219 : }
2220 :
2221 23 : if (i == 0) {
2222 5 : stream << command;
2223 : } else {
2224 18 : stream << "," << command;
2225 : }
2226 23 : ++i;
2227 : }
2228 5 : stream << "}";
2229 : }
2230 :
2231 40 : return stream.str();
2232 : }
2233 :
2234 : // Printing the one and only help message
2235 : // I've stuck with a simple message format, nothing fancy.
2236 : [[deprecated("Use cout << program; instead. See also help().")]] std::string
2237 : print_help() const {
2238 : auto out = help();
2239 : std::cout << out.rdbuf();
2240 : return out.str();
2241 : }
2242 :
2243 595 : void add_subparser(ArgumentParser &parser) {
2244 595 : parser.m_parser_path = m_program_name + " " + parser.m_program_name;
2245 595 : auto it = m_subparsers.emplace(std::cend(m_subparsers), parser);
2246 595 : m_subparser_map.insert_or_assign(parser.m_program_name, it);
2247 595 : m_subparser_used.insert_or_assign(parser.m_program_name, false);
2248 595 : }
2249 :
2250 : void set_suppress(bool suppress) { m_suppress = suppress; }
2251 :
2252 : protected:
2253 558 : const MutuallyExclusiveGroup *get_belonging_mutex(const Argument *arg) const {
2254 1899 : for (const auto &mutex : m_mutually_exclusive_groups) {
2255 1410 : if (std::find(mutex.m_elements.begin(), mutex.m_elements.end(), arg) !=
2256 2820 : mutex.m_elements.end()) {
2257 69 : return &mutex;
2258 : }
2259 : }
2260 489 : return nullptr;
2261 : }
2262 :
2263 20 : bool is_valid_prefix_char(char c) const {
2264 20 : return m_prefix_chars.find(c) != std::string::npos;
2265 : }
2266 :
2267 0 : char get_any_valid_prefix_char() const { return m_prefix_chars[0]; }
2268 :
2269 : /*
2270 : * Pre-process this argument list. Anything starting with "--", that
2271 : * contains an =, where the prefix before the = has an entry in the
2272 : * options table, should be split.
2273 : */
2274 : std::vector<std::string>
2275 11006 : preprocess_arguments(const std::vector<std::string> &raw_arguments) const {
2276 11006 : std::vector<std::string> arguments{};
2277 75911 : for (const auto &arg : raw_arguments) {
2278 :
2279 : const auto argument_starts_with_prefix_chars =
2280 86929 : [this](const std::string &a) -> bool {
2281 43466 : if (!a.empty()) {
2282 :
2283 82963 : const auto legal_prefix = [this](char c) -> bool {
2284 82963 : return m_prefix_chars.find(c) != std::string::npos;
2285 43463 : };
2286 :
2287 : // Windows-style
2288 : // if '/' is a legal prefix char
2289 : // then allow single '/' followed by argument name, followed by an
2290 : // assign char, e.g., ':' e.g., 'test.exe /A:Foo'
2291 43463 : const auto windows_style = legal_prefix('/');
2292 :
2293 43463 : if (windows_style) {
2294 0 : if (legal_prefix(a[0])) {
2295 38094 : return true;
2296 : }
2297 : } else {
2298 : // Slash '/' is not a legal prefix char
2299 : // For all other characters, only support long arguments
2300 : // i.e., the argument must start with 2 prefix chars, e.g,
2301 : // '--foo' e,g, './test --foo=Bar -DARG=yes'
2302 43463 : if (a.size() > 1) {
2303 38094 : return (legal_prefix(a[0]) && legal_prefix(a[1]));
2304 : }
2305 : }
2306 : }
2307 5372 : return false;
2308 64905 : };
2309 :
2310 : // Check that:
2311 : // - We don't have an argument named exactly this
2312 : // - The argument starts with a prefix char, e.g., "--"
2313 : // - The argument contains an assign char, e.g., "="
2314 64905 : auto assign_char_pos = arg.find_first_of(m_assign_chars);
2315 :
2316 108371 : if (m_argument_map.find(arg) == m_argument_map.end() &&
2317 108371 : argument_starts_with_prefix_chars(arg) &&
2318 : assign_char_pos != std::string::npos) {
2319 : // Get the name of the potential option, and check it exists
2320 5 : std::string opt_name = arg.substr(0, assign_char_pos);
2321 5 : if (m_argument_map.find(opt_name) != m_argument_map.end()) {
2322 : // This is the name of an option! Split it into two parts
2323 5 : arguments.push_back(std::move(opt_name));
2324 5 : arguments.push_back(arg.substr(assign_char_pos + 1));
2325 5 : continue;
2326 : }
2327 : }
2328 : // If we've fallen through to here, then it's a standard argument
2329 64900 : arguments.push_back(arg);
2330 : }
2331 11006 : return arguments;
2332 : }
2333 :
2334 : /*
2335 : * @throws std::runtime_error in case of any invalid argument
2336 : */
2337 4991 : void parse_args_internal(const std::vector<std::string> &raw_arguments) {
2338 5022 : auto arguments = preprocess_arguments(raw_arguments);
2339 4991 : if (m_program_name.empty() && !arguments.empty()) {
2340 0 : m_program_name = arguments.front();
2341 : }
2342 4991 : auto end = std::end(arguments);
2343 4991 : auto positional_argument_it = std::begin(m_positional_arguments);
2344 16263 : for (auto it = std::next(std::begin(arguments)); it != end;) {
2345 11303 : const auto ¤t_argument = *it;
2346 11303 : if (Argument::is_positional(current_argument, m_prefix_chars)) {
2347 1329 : if (positional_argument_it == std::end(m_positional_arguments)) {
2348 :
2349 : // Check sub-parsers
2350 0 : auto subparser_it = m_subparser_map.find(current_argument);
2351 0 : if (subparser_it != m_subparser_map.end()) {
2352 :
2353 : // build list of remaining args
2354 : const auto unprocessed_arguments =
2355 0 : std::vector<std::string>(it, end);
2356 :
2357 : // invoke subparser
2358 0 : m_is_parsed = true;
2359 0 : m_subparser_used[current_argument] = true;
2360 0 : return subparser_it->second->get().parse_args(
2361 0 : unprocessed_arguments);
2362 : }
2363 :
2364 0 : if (m_positional_arguments.empty()) {
2365 :
2366 : // Ask the user if they argument they provided was a typo
2367 : // for some sub-parser,
2368 : // e.g., user provided `git totes` instead of `git notes`
2369 0 : if (!m_subparser_map.empty()) {
2370 : throw std::runtime_error(
2371 0 : "Failed to parse '" + current_argument + "', did you mean '" +
2372 0 : std::string{details::get_most_similar_string(
2373 0 : m_subparser_map, current_argument)} +
2374 0 : "'");
2375 : }
2376 :
2377 : // Ask the user if they meant to use a specific optional argument
2378 0 : if (!m_optional_arguments.empty()) {
2379 0 : for (const auto &opt : m_optional_arguments) {
2380 0 : if (!opt.m_implicit_value.has_value()) {
2381 : // not a flag, requires a value
2382 0 : if (!opt.m_is_used) {
2383 : throw std::runtime_error(
2384 0 : "Zero positional arguments expected, did you mean " +
2385 0 : opt.get_usage_full());
2386 : }
2387 : }
2388 : }
2389 :
2390 0 : throw std::runtime_error("Zero positional arguments expected");
2391 : } else {
2392 0 : throw std::runtime_error("Zero positional arguments expected");
2393 : }
2394 : } else {
2395 : throw std::runtime_error("Maximum number of positional arguments "
2396 0 : "exceeded, failed to parse '" +
2397 0 : current_argument + "'");
2398 : }
2399 : }
2400 1329 : auto argument = positional_argument_it++;
2401 :
2402 : // Deal with the situation of <positional_arg1>... <positional_arg2>
2403 2525 : if (argument->m_num_args_range.get_min() == 1 &&
2404 1302 : argument->m_num_args_range.get_max() == (std::numeric_limits<std::size_t>::max)() &&
2405 206 : positional_argument_it != std::end(m_positional_arguments) &&
2406 200 : std::next(positional_argument_it) == std::end(m_positional_arguments) &&
2407 2625 : positional_argument_it->m_num_args_range.get_min() == 1 &&
2408 100 : positional_argument_it->m_num_args_range.get_max() == 1 ) {
2409 100 : if (std::next(it) != end) {
2410 100 : positional_argument_it->consume(std::prev(end), end);
2411 100 : end = std::prev(end);
2412 : } else {
2413 0 : throw std::runtime_error("Missing " + positional_argument_it->m_names.front());
2414 : }
2415 : }
2416 :
2417 1329 : it = argument->consume(it, end);
2418 1329 : continue;
2419 : }
2420 :
2421 9974 : auto arg_map_it = m_argument_map.find(current_argument);
2422 9974 : if (arg_map_it != m_argument_map.end()) {
2423 9974 : auto argument = arg_map_it->second;
2424 9974 : it = argument->consume(std::next(it), end, arg_map_it->first);
2425 0 : } else if (const auto &compound_arg = current_argument;
2426 0 : compound_arg.size() > 1 &&
2427 0 : is_valid_prefix_char(compound_arg[0]) &&
2428 0 : !is_valid_prefix_char(compound_arg[1])) {
2429 0 : ++it;
2430 0 : for (std::size_t j = 1; j < compound_arg.size(); j++) {
2431 0 : auto hypothetical_arg = std::string{'-', compound_arg[j]};
2432 0 : auto arg_map_it2 = m_argument_map.find(hypothetical_arg);
2433 0 : if (arg_map_it2 != m_argument_map.end()) {
2434 0 : auto argument = arg_map_it2->second;
2435 0 : it = argument->consume(it, end, arg_map_it2->first);
2436 : } else {
2437 0 : throw std::runtime_error("Unknown argument: " + current_argument);
2438 : }
2439 : }
2440 : } else {
2441 0 : throw std::runtime_error("Unknown argument: " + current_argument);
2442 : }
2443 : }
2444 4960 : m_is_parsed = true;
2445 : }
2446 :
2447 : /*
2448 : * Like parse_args_internal but collects unused args into a vector<string>
2449 : */
2450 : std::vector<std::string>
2451 : parse_known_args_internal(const std::vector<std::string> &raw_arguments) {
2452 : auto arguments = preprocess_arguments(raw_arguments);
2453 :
2454 : std::vector<std::string> unknown_arguments{};
2455 :
2456 : if (m_program_name.empty() && !arguments.empty()) {
2457 : m_program_name = arguments.front();
2458 : }
2459 : auto end = std::end(arguments);
2460 : auto positional_argument_it = std::begin(m_positional_arguments);
2461 : for (auto it = std::next(std::begin(arguments)); it != end;) {
2462 : const auto ¤t_argument = *it;
2463 : if (Argument::is_positional(current_argument, m_prefix_chars)) {
2464 : if (positional_argument_it == std::end(m_positional_arguments)) {
2465 :
2466 : // Check sub-parsers
2467 : auto subparser_it = m_subparser_map.find(current_argument);
2468 : if (subparser_it != m_subparser_map.end()) {
2469 :
2470 : // build list of remaining args
2471 : const auto unprocessed_arguments =
2472 : std::vector<std::string>(it, end);
2473 :
2474 : // invoke subparser
2475 : m_is_parsed = true;
2476 : m_subparser_used[current_argument] = true;
2477 : return subparser_it->second->get().parse_known_args_internal(
2478 : unprocessed_arguments);
2479 : }
2480 :
2481 : // save current argument as unknown and go to next argument
2482 : unknown_arguments.push_back(current_argument);
2483 : ++it;
2484 : } else {
2485 : // current argument is the value of a positional argument
2486 : // consume it
2487 : auto argument = positional_argument_it++;
2488 : it = argument->consume(it, end);
2489 : }
2490 : continue;
2491 : }
2492 :
2493 : auto arg_map_it = m_argument_map.find(current_argument);
2494 : if (arg_map_it != m_argument_map.end()) {
2495 : auto argument = arg_map_it->second;
2496 : it = argument->consume(std::next(it), end, arg_map_it->first);
2497 : } else if (const auto &compound_arg = current_argument;
2498 : compound_arg.size() > 1 &&
2499 : is_valid_prefix_char(compound_arg[0]) &&
2500 : !is_valid_prefix_char(compound_arg[1])) {
2501 : ++it;
2502 : for (std::size_t j = 1; j < compound_arg.size(); j++) {
2503 : auto hypothetical_arg = std::string{'-', compound_arg[j]};
2504 : auto arg_map_it2 = m_argument_map.find(hypothetical_arg);
2505 : if (arg_map_it2 != m_argument_map.end()) {
2506 : auto argument = arg_map_it2->second;
2507 : it = argument->consume(it, end, arg_map_it2->first);
2508 : } else {
2509 : unknown_arguments.push_back(current_argument);
2510 : break;
2511 : }
2512 : }
2513 : } else {
2514 : // current argument is an optional-like argument that is unknown
2515 : // save it and move to next argument
2516 : unknown_arguments.push_back(current_argument);
2517 : ++it;
2518 : }
2519 : }
2520 : m_is_parsed = true;
2521 : return unknown_arguments;
2522 : }
2523 :
2524 : // Used by print_help.
2525 0 : std::size_t get_length_of_longest_argument() const {
2526 0 : if (m_argument_map.empty()) {
2527 0 : return 0;
2528 : }
2529 0 : std::size_t max_size = 0;
2530 0 : for ([[maybe_unused]] const auto &[unused, argument] : m_argument_map) {
2531 0 : max_size =
2532 0 : std::max<std::size_t>(max_size, argument->get_arguments_length());
2533 : }
2534 0 : for ([[maybe_unused]] const auto &[command, unused] : m_subparser_map) {
2535 0 : max_size = std::max<std::size_t>(max_size, command.size());
2536 : }
2537 0 : return max_size;
2538 : }
2539 :
2540 : using argument_it = std::list<Argument>::iterator;
2541 : using mutex_group_it = std::vector<MutuallyExclusiveGroup>::iterator;
2542 : using argument_parser_it =
2543 : std::list<std::reference_wrapper<ArgumentParser>>::iterator;
2544 :
2545 243420 : void index_argument(argument_it it) {
2546 495278 : for (const auto &name : std::as_const(it->m_names)) {
2547 251858 : m_argument_map.insert_or_assign(name, it);
2548 : }
2549 243420 : }
2550 :
2551 : std::string m_program_name;
2552 : std::string m_version;
2553 : std::string m_description;
2554 : std::string m_epilog;
2555 : bool m_exit_on_default_arguments = true;
2556 : std::string m_prefix_chars{"-"};
2557 : std::string m_assign_chars{"="};
2558 : bool m_is_parsed = false;
2559 : std::list<Argument> m_positional_arguments;
2560 : std::list<Argument> m_optional_arguments;
2561 : std::map<std::string, argument_it> m_argument_map;
2562 : std::string m_parser_path;
2563 : std::list<std::reference_wrapper<ArgumentParser>> m_subparsers;
2564 : std::map<std::string, argument_parser_it> m_subparser_map;
2565 : std::map<std::string, bool> m_subparser_used;
2566 : std::vector<MutuallyExclusiveGroup> m_mutually_exclusive_groups;
2567 : bool m_suppress = false;
2568 : std::size_t m_usage_max_line_width = std::numeric_limits<std::size_t>::max();
2569 : bool m_usage_break_on_mutex = false;
2570 : int m_usage_newline_counter = 0;
2571 : std::vector<std::string> m_group_names;
2572 : };
2573 :
2574 : } // namespace argparse
|