CLI11
C++11 Command Line Interface Parser
Loading...
Searching...
No Matches
Option_inl.hpp
1// Copyright (c) 2017-2024, University of Cincinnati, developed by Henry Schreiner
2// under NSF AWARD 1414736 and by the respective contributors.
3// All rights reserved.
4//
5// SPDX-License-Identifier: BSD-3-Clause
6
7#pragma once
8
9// IWYU pragma: private, include "CLI/CLI.hpp"
10
11// This include is only needed for IDEs to discover symbols
12#include "../Option.hpp"
13
14// [CLI11:public_includes:set]
15#include <algorithm>
16#include <string>
17#include <utility>
18#include <vector>
19// [CLI11:public_includes:end]
20
21namespace CLI {
22// [CLI11:option_inl_hpp:verbatim]
23
24template <typename CRTP> template <typename T> void OptionBase<CRTP>::copy_to(T *other) const {
25 other->group(group_);
26 other->required(required_);
27 other->ignore_case(ignore_case_);
28 other->ignore_underscore(ignore_underscore_);
29 other->configurable(configurable_);
30 other->disable_flag_override(disable_flag_override_);
31 other->delimiter(delimiter_);
32 other->always_capture_default(always_capture_default_);
33 other->multi_option_policy(multi_option_policy_);
34}
35
36CLI11_INLINE Option *Option::expected(int value) {
37 if(value < 0) {
38 expected_min_ = -value;
41 }
42 allow_extra_args_ = true;
43 flag_like_ = false;
44 } else if(value == detail::expected_max_vector_size) {
45 expected_min_ = 1;
46 expected_max_ = detail::expected_max_vector_size;
47 allow_extra_args_ = true;
48 flag_like_ = false;
49 } else {
50 expected_min_ = value;
51 expected_max_ = value;
53 }
54 return this;
55}
56
57CLI11_INLINE Option *Option::expected(int value_min, int value_max) {
58 if(value_min < 0) {
59 value_min = -value_min;
60 }
61
62 if(value_max < 0) {
63 value_max = detail::expected_max_vector_size;
64 }
65 if(value_max < value_min) {
66 expected_min_ = value_max;
67 expected_max_ = value_min;
68 } else {
69 expected_max_ = value_max;
70 expected_min_ = value_min;
71 }
72
73 return this;
74}
75
76CLI11_INLINE Option *Option::check(Validator validator, const std::string &validator_name) {
77 validator.non_modifying();
78 validators_.push_back(std::move(validator));
79 if(!validator_name.empty())
80 validators_.back().name(validator_name);
81 return this;
82}
83
84CLI11_INLINE Option *Option::check(std::function<std::string(const std::string &)> Validator,
85 std::string Validator_description,
86 std::string Validator_name) {
87 validators_.emplace_back(Validator, std::move(Validator_description), std::move(Validator_name));
88 validators_.back().non_modifying();
89 return this;
90}
91
92CLI11_INLINE Option *Option::transform(Validator Validator, const std::string &Validator_name) {
93 validators_.insert(validators_.begin(), std::move(Validator));
94 if(!Validator_name.empty())
95 validators_.front().name(Validator_name);
96 return this;
97}
98
99CLI11_INLINE Option *Option::transform(const std::function<std::string(std::string)> &func,
100 std::string transform_description,
101 std::string transform_name) {
102 validators_.insert(validators_.begin(),
103 Validator(
104 [func](std::string &val) {
105 val = func(val);
106 return std::string{};
107 },
108 std::move(transform_description),
109 std::move(transform_name)));
110
111 return this;
112}
113
114CLI11_INLINE Option *Option::each(const std::function<void(std::string)> &func) {
115 validators_.emplace_back(
116 [func](std::string &inout) {
117 func(inout);
118 return std::string{};
119 },
120 std::string{});
121 return this;
122}
123
124CLI11_INLINE Validator *Option::get_validator(const std::string &Validator_name) {
125 for(auto &Validator : validators_) {
126 if(Validator_name == Validator.get_name()) {
127 return &Validator;
128 }
129 }
130 if((Validator_name.empty()) && (!validators_.empty())) {
131 return &(validators_.front());
132 }
133 throw OptionNotFound(std::string{"Validator "} + Validator_name + " Not Found");
134}
135
136CLI11_INLINE Validator *Option::get_validator(int index) {
137 // This is an signed int so that it is not equivalent to a pointer.
138 if(index >= 0 && index < static_cast<int>(validators_.size())) {
139 return &(validators_[static_cast<decltype(validators_)::size_type>(index)]);
140 }
141 throw OptionNotFound("Validator index is not valid");
142}
143
144CLI11_INLINE bool Option::remove_needs(Option *opt) {
145 auto iterator = std::find(std::begin(needs_), std::end(needs_), opt);
146
147 if(iterator == std::end(needs_)) {
148 return false;
149 }
150 needs_.erase(iterator);
151 return true;
152}
153
154CLI11_INLINE Option *Option::excludes(Option *opt) {
155 if(opt == this) {
156 throw(IncorrectConstruction("and option cannot exclude itself"));
157 }
158 excludes_.insert(opt);
159
160 // Help text should be symmetric - excluding a should exclude b
161 opt->excludes_.insert(this);
162
163 // Ignoring the insert return value, excluding twice is now allowed.
164 // (Mostly to allow both directions to be excluded by user, even though the library does it for you.)
165
166 return this;
167}
168
169CLI11_INLINE bool Option::remove_excludes(Option *opt) {
170 auto iterator = std::find(std::begin(excludes_), std::end(excludes_), opt);
171
172 if(iterator == std::end(excludes_)) {
173 return false;
174 }
175 excludes_.erase(iterator);
176 return true;
177}
178
179template <typename T> Option *Option::ignore_case(bool value) {
180 if(!ignore_case_ && value) {
181 ignore_case_ = value;
182 auto *parent = static_cast<T *>(parent_);
183 for(const Option_p &opt : parent->options_) {
184 if(opt.get() == this) {
185 continue;
186 }
187 const auto &omatch = opt->matching_name(*this);
188 if(!omatch.empty()) {
189 ignore_case_ = false;
190 throw OptionAlreadyAdded("adding ignore case caused a name conflict with " + omatch);
191 }
192 }
193 } else {
194 ignore_case_ = value;
195 }
196 return this;
197}
198
199template <typename T> Option *Option::ignore_underscore(bool value) {
200
201 if(!ignore_underscore_ && value) {
202 ignore_underscore_ = value;
203 auto *parent = static_cast<T *>(parent_);
204 for(const Option_p &opt : parent->options_) {
205 if(opt.get() == this) {
206 continue;
207 }
208 const auto &omatch = opt->matching_name(*this);
209 if(!omatch.empty()) {
210 ignore_underscore_ = false;
211 throw OptionAlreadyAdded("adding ignore underscore caused a name conflict with " + omatch);
212 }
213 }
214 } else {
215 ignore_underscore_ = value;
216 }
217 return this;
218}
219
220CLI11_INLINE Option *Option::multi_option_policy(MultiOptionPolicy value) {
221 if(value != multi_option_policy_) {
222 if(multi_option_policy_ == MultiOptionPolicy::Throw && expected_max_ == detail::expected_max_vector_size &&
223 expected_min_ > 1) { // this bizarre condition is to maintain backwards compatibility
224 // with the previous behavior of expected_ with vectors
226 }
227 multi_option_policy_ = value;
229 }
230 return this;
231}
232
233CLI11_NODISCARD CLI11_INLINE std::string Option::get_name(bool positional, bool all_options) const {
234 if(get_group().empty())
235 return {}; // Hidden
236
237 if(all_options) {
238
239 std::vector<std::string> name_list;
240
242 if((positional && (!pname_.empty())) || (snames_.empty() && lnames_.empty())) {
243 name_list.push_back(pname_);
244 }
245 if((get_items_expected() == 0) && (!fnames_.empty())) {
246 for(const std::string &sname : snames_) {
247 name_list.push_back("-" + sname);
248 if(check_fname(sname)) {
249 name_list.back() += "{" + get_flag_value(sname, "") + "}";
250 }
251 }
252
253 for(const std::string &lname : lnames_) {
254 name_list.push_back("--" + lname);
255 if(check_fname(lname)) {
256 name_list.back() += "{" + get_flag_value(lname, "") + "}";
257 }
258 }
259 } else {
260 for(const std::string &sname : snames_)
261 name_list.push_back("-" + sname);
262
263 for(const std::string &lname : lnames_)
264 name_list.push_back("--" + lname);
265 }
266
267 return detail::join(name_list);
268 }
269
270 // This returns the positional name no matter what
271 if(positional)
272 return pname_;
273
274 // Prefer long name
275 if(!lnames_.empty())
276 return std::string(2, '-') + lnames_[0];
277
278 // Or short name if no long name
279 if(!snames_.empty())
280 return std::string(1, '-') + snames_[0];
281
282 // If positional is the only name, it's okay to use that
283 return pname_;
284}
285
286CLI11_INLINE void Option::run_callback() {
287 if(force_callback_ && results_.empty()) {
289 }
291 _validate_results(results_);
293 }
294
296 _reduce_results(proc_results_, results_);
298 }
301 if(!(callback_)) {
302 return;
303 }
304 const results_t &send_results = proc_results_.empty() ? results_ : proc_results_;
305 bool local_result = callback_(send_results);
306
307 if(!local_result)
309 }
310}
311
312CLI11_NODISCARD CLI11_INLINE const std::string &Option::matching_name(const Option &other) const {
313 static const std::string estring;
314 for(const std::string &sname : snames_) {
315 if(other.check_sname(sname))
316 return sname;
317 if(other.check_lname(sname))
318 return sname;
319 }
320 for(const std::string &lname : lnames_) {
321 if(other.check_lname(lname))
322 return lname;
323 if(lname.size() == 1) {
324 if(other.check_sname(lname)) {
325 return lname;
326 }
327 }
328 }
329 if(snames_.empty() && lnames_.empty() && !pname_.empty()) {
330 if(other.check_sname(pname_) || other.check_lname(pname_) || pname_ == other.pname_)
331 return pname_;
332 }
333 if(other.snames_.empty() && other.fnames_.empty() && !other.pname_.empty()) {
334 if(check_sname(other.pname_) || check_lname(other.pname_) || (pname_ == other.pname_))
335 return other.pname_;
336 }
337 if(ignore_case_ ||
338 ignore_underscore_) { // We need to do the inverse, in case we are ignore_case or ignore underscore
339 for(const std::string &sname : other.snames_)
340 if(check_sname(sname))
341 return sname;
342 for(const std::string &lname : other.lnames_)
343 if(check_lname(lname))
344 return lname;
345 }
346 return estring;
347}
348
349CLI11_NODISCARD CLI11_INLINE bool Option::check_name(const std::string &name) const {
350
351 if(name.length() > 2 && name[0] == '-' && name[1] == '-')
352 return check_lname(name.substr(2));
353 if(name.length() > 1 && name.front() == '-')
354 return check_sname(name.substr(1));
355 if(!pname_.empty()) {
356 std::string local_pname = pname_;
357 std::string local_name = name;
359 local_pname = detail::remove_underscore(local_pname);
360 local_name = detail::remove_underscore(local_name);
361 }
362 if(ignore_case_) {
363 local_pname = detail::to_lower(local_pname);
364 local_name = detail::to_lower(local_name);
365 }
366 if(local_name == local_pname) {
367 return true;
368 }
369 }
370
371 if(!envname_.empty()) {
372 // this needs to be the original since envname_ shouldn't match on case insensitivity
373 return (name == envname_);
374 }
375 return false;
376}
377
378CLI11_NODISCARD CLI11_INLINE std::string Option::get_flag_value(const std::string &name,
379 std::string input_value) const {
380 static const std::string trueString{"true"};
381 static const std::string falseString{"false"};
382 static const std::string emptyString{"{}"};
383 // check for disable flag override_
385 if(!((input_value.empty()) || (input_value == emptyString))) {
386 auto default_ind = detail::find_member(name, fnames_, ignore_case_, ignore_underscore_);
387 if(default_ind >= 0) {
388 // We can static cast this to std::size_t because it is more than 0 in this block
389 if(default_flag_values_[static_cast<std::size_t>(default_ind)].second != input_value) {
390 if(input_value == default_str_ && force_callback_) {
391 return input_value;
392 }
393 throw(ArgumentMismatch::FlagOverride(name));
394 }
395 } else {
396 if(input_value != trueString) {
397 throw(ArgumentMismatch::FlagOverride(name));
398 }
399 }
400 }
401 }
402 auto ind = detail::find_member(name, fnames_, ignore_case_, ignore_underscore_);
403 if((input_value.empty()) || (input_value == emptyString)) {
404 if(flag_like_) {
405 return (ind < 0) ? trueString : default_flag_values_[static_cast<std::size_t>(ind)].second;
406 }
407 return (ind < 0) ? default_str_ : default_flag_values_[static_cast<std::size_t>(ind)].second;
408 }
409 if(ind < 0) {
410 return input_value;
411 }
412 if(default_flag_values_[static_cast<std::size_t>(ind)].second == falseString) {
413 errno = 0;
414 auto val = detail::to_flag_value(input_value);
415 if(errno != 0) {
416 errno = 0;
417 return input_value;
418 }
419 return (val == 1) ? falseString : (val == (-1) ? trueString : std::to_string(-val));
420 }
421 return input_value;
422}
423
424CLI11_INLINE Option *Option::add_result(std::string s) {
425 _add_result(std::move(s), results_);
427 return this;
428}
429
430CLI11_INLINE Option *Option::add_result(std::string s, int &results_added) {
431 results_added = _add_result(std::move(s), results_);
433 return this;
434}
435
436CLI11_INLINE Option *Option::add_result(std::vector<std::string> s) {
438 for(auto &str : s) {
439 _add_result(std::move(str), results_);
440 }
441 return this;
442}
443
444CLI11_NODISCARD CLI11_INLINE results_t Option::reduced_results() const {
445 results_t res = proc_results_.empty() ? results_ : proc_results_;
448 res = results_;
449 _validate_results(res);
450 }
451 if(!res.empty()) {
452 results_t extra;
453 _reduce_results(extra, res);
454 if(!extra.empty()) {
455 res = std::move(extra);
456 }
457 }
458 }
459 return res;
460}
461
462CLI11_INLINE Option *Option::type_size(int option_type_size) {
463 if(option_type_size < 0) {
464 // this section is included for backwards compatibility
465 type_size_max_ = -option_type_size;
466 type_size_min_ = -option_type_size;
467 expected_max_ = detail::expected_max_vector_size;
468 } else {
469 type_size_max_ = option_type_size;
470 if(type_size_max_ < detail::expected_max_vector_size) {
471 type_size_min_ = option_type_size;
472 } else {
473 inject_separator_ = true;
474 }
475 if(type_size_max_ == 0)
476 required_ = false;
477 }
478 return this;
479}
480
481CLI11_INLINE Option *Option::type_size(int option_type_size_min, int option_type_size_max) {
482 if(option_type_size_min < 0 || option_type_size_max < 0) {
483 // this section is included for backwards compatibility
484 expected_max_ = detail::expected_max_vector_size;
485 option_type_size_min = (std::abs)(option_type_size_min);
486 option_type_size_max = (std::abs)(option_type_size_max);
487 }
488
489 if(option_type_size_min > option_type_size_max) {
490 type_size_max_ = option_type_size_min;
491 type_size_min_ = option_type_size_max;
492 } else {
493 type_size_min_ = option_type_size_min;
494 type_size_max_ = option_type_size_max;
495 }
496 if(type_size_max_ == 0) {
497 required_ = false;
498 }
499 if(type_size_max_ >= detail::expected_max_vector_size) {
500 inject_separator_ = true;
501 }
502 return this;
503}
504
505CLI11_NODISCARD CLI11_INLINE std::string Option::get_type_name() const {
506 std::string full_type_name = type_name_();
507 if(!validators_.empty()) {
508 for(const auto &Validator : validators_) {
509 std::string vtype = Validator.get_description();
510 if(!vtype.empty()) {
511 full_type_name += ":" + vtype;
512 }
513 }
514 }
515 return full_type_name;
516}
517
518CLI11_INLINE void Option::_validate_results(results_t &res) const {
519 // Run the Validators (can change the string)
520 if(!validators_.empty()) {
521 if(type_size_max_ > 1) { // in this context index refers to the index in the type
522 int index = 0;
523 if(get_items_expected_max() < static_cast<int>(res.size()) &&
524 (multi_option_policy_ == CLI::MultiOptionPolicy::TakeLast ||
525 multi_option_policy_ == CLI::MultiOptionPolicy::Reverse)) {
526 // create a negative index for the earliest ones
527 index = get_items_expected_max() - static_cast<int>(res.size());
528 }
529
530 for(std::string &result : res) {
531 if(detail::is_separator(result) && type_size_max_ != type_size_min_ && index >= 0) {
532 index = 0; // reset index for variable size chunks
533 continue;
534 }
535 auto err_msg = _validate(result, (index >= 0) ? (index % type_size_max_) : index);
536 if(!err_msg.empty())
537 throw ValidationError(get_name(), err_msg);
538 ++index;
539 }
540 } else {
541 int index = 0;
542 if(expected_max_ < static_cast<int>(res.size()) &&
543 (multi_option_policy_ == CLI::MultiOptionPolicy::TakeLast ||
544 multi_option_policy_ == CLI::MultiOptionPolicy::Reverse)) {
545 // create a negative index for the earliest ones
546 index = expected_max_ - static_cast<int>(res.size());
547 }
548 for(std::string &result : res) {
549 auto err_msg = _validate(result, index);
550 ++index;
551 if(!err_msg.empty())
552 throw ValidationError(get_name(), err_msg);
553 }
554 }
555 }
556}
557
558CLI11_INLINE void Option::_reduce_results(results_t &out, const results_t &original) const {
559
560 // max num items expected or length of vector, always at least 1
561 // Only valid for a trimming policy
562
563 out.clear();
564 // Operation depends on the policy setting
565 switch(multi_option_policy_) {
566 case MultiOptionPolicy::TakeAll:
567 break;
568 case MultiOptionPolicy::TakeLast: {
569 // Allow multi-option sizes (including 0)
570 std::size_t trim_size = std::min<std::size_t>(
571 static_cast<std::size_t>(std::max<int>(get_items_expected_max(), 1)), original.size());
572 if(original.size() != trim_size) {
573 out.assign(original.end() - static_cast<results_t::difference_type>(trim_size), original.end());
574 }
575 } break;
576 case MultiOptionPolicy::Reverse: {
577 // Allow multi-option sizes (including 0)
578 std::size_t trim_size = std::min<std::size_t>(
579 static_cast<std::size_t>(std::max<int>(get_items_expected_max(), 1)), original.size());
580 if(original.size() != trim_size || trim_size > 1) {
581 out.assign(original.end() - static_cast<results_t::difference_type>(trim_size), original.end());
582 }
583 std::reverse(out.begin(), out.end());
584 } break;
585 case MultiOptionPolicy::TakeFirst: {
586 std::size_t trim_size = std::min<std::size_t>(
587 static_cast<std::size_t>(std::max<int>(get_items_expected_max(), 1)), original.size());
588 if(original.size() != trim_size) {
589 out.assign(original.begin(), original.begin() + static_cast<results_t::difference_type>(trim_size));
590 }
591 } break;
592 case MultiOptionPolicy::Join:
593 if(results_.size() > 1) {
594 out.push_back(detail::join(original, std::string(1, (delimiter_ == '\0') ? '\n' : delimiter_)));
595 }
596 break;
597 case MultiOptionPolicy::Sum:
598 out.push_back(detail::sum_string_vector(original));
599 break;
600 case MultiOptionPolicy::Throw:
601 default: {
602 auto num_min = static_cast<std::size_t>(get_items_expected_min());
603 auto num_max = static_cast<std::size_t>(get_items_expected_max());
604 if(num_min == 0) {
605 num_min = 1;
606 }
607 if(num_max == 0) {
608 num_max = 1;
609 }
610 if(original.size() < num_min) {
611 throw ArgumentMismatch::AtLeast(get_name(), static_cast<int>(num_min), original.size());
612 }
613 if(original.size() > num_max) {
614 if(original.size() == 2 && num_max == 1 && original[1] == "%%" && original[0] == "{}") {
615 // this condition is a trap for the following empty indicator check on config files
616 out = original;
617 } else {
618 throw ArgumentMismatch::AtMost(get_name(), static_cast<int>(num_max), original.size());
619 }
620 }
621 break;
622 }
623 }
624 // this check is to allow an empty vector in certain circumstances but not if expected is not zero.
625 // {} is the indicator for an empty container
626 if(out.empty()) {
627 if(original.size() == 1 && original[0] == "{}" && get_items_expected_min() > 0) {
628 out.emplace_back("{}");
629 out.emplace_back("%%");
630 }
631 } else if(out.size() == 1 && out[0] == "{}" && get_items_expected_min() > 0) {
632 out.emplace_back("%%");
633 }
634}
635
636CLI11_INLINE std::string Option::_validate(std::string &result, int index) const {
637 std::string err_msg;
638 if(result.empty() && expected_min_ == 0) {
639 // an empty with nothing expected is allowed
640 return err_msg;
641 }
642 for(const auto &vali : validators_) {
643 auto v = vali.get_application_index();
644 if(v == -1 || v == index) {
645 try {
646 err_msg = vali(result);
647 } catch(const ValidationError &err) {
648 err_msg = err.what();
649 }
650 if(!err_msg.empty())
651 break;
652 }
653 }
654
655 return err_msg;
656}
657
658CLI11_INLINE int Option::_add_result(std::string &&result, std::vector<std::string> &res) const {
659 int result_count = 0;
660 if(allow_extra_args_ && !result.empty() && result.front() == '[' &&
661 result.back() == ']') { // this is now a vector string likely from the default or user entry
662 result.pop_back();
663
664 for(auto &var : CLI::detail::split(result.substr(1), ',')) {
665 if(!var.empty()) {
666 result_count += _add_result(std::move(var), res);
667 }
668 }
669 return result_count;
670 }
671 if(delimiter_ == '\0') {
672 res.push_back(std::move(result));
673 ++result_count;
674 } else {
675 if((result.find_first_of(delimiter_) != std::string::npos)) {
676 for(const auto &var : CLI::detail::split(result, delimiter_)) {
677 if(!var.empty()) {
678 res.push_back(var);
679 ++result_count;
680 }
681 }
682 } else {
683 res.push_back(std::move(result));
684 ++result_count;
685 }
686 }
687 return result_count;
688}
689// [CLI11:option_inl_hpp:end]
690} // namespace CLI
Thrown when conversion call back fails, such as when an int fails to coerce to a string.
Definition Error.hpp:205
Thrown when an option is set to conflicting values (non-vector and multi args, for example)
Definition Error.hpp:96
Thrown when an option already exists.
Definition Error.hpp:144
MultiOptionPolicy multi_option_policy_
Policy for handling multiple arguments beyond the expected Max.
Definition Option.hpp:81
bool ignore_case_
Ignore the case when matching (option, not value)
Definition Option.hpp:63
bool disable_flag_override_
Disable overriding flag values with '=value'.
Definition Option.hpp:72
bool required_
True if this is a required option.
Definition Option.hpp:60
char delimiter_
Specify a delimiter character for vector arguments.
Definition Option.hpp:75
bool ignore_underscore_
Ignore underscores when matching (option, not value)
Definition Option.hpp:66
CLI11_NODISCARD const std::string & get_group() const
Get the group of this option.
Definition Option.hpp:115
void copy_to(T *other) const
Copy the contents to another similar class (one based on OptionBase)
Definition Option_inl.hpp:24
Definition Option.hpp:231
Option * type_size(int option_type_size)
Set a custom option size.
Definition Option_inl.hpp:462
Option * expected(int value)
Set the number of expected arguments.
Definition Option_inl.hpp:36
std::string default_str_
A human readable default value, either manually set, captured, or captured by default.
Definition Option.hpp:265
CLI11_NODISCARD bool check_name(const std::string &name) const
Check a name. Requires "-" or "--" for short / long, supports positional name.
Definition Option_inl.hpp:349
std::function< std::string()> type_name_
Definition Option.hpp:273
@ reduced
a subset of results has been generated
@ callback_run
the callback has been executed
@ validated
the results have been validated
@ parsing
The option is currently collecting parsed results.
option_state current_option_state_
Whether the callback has run (needed for INI parsing)
Definition Option.hpp:328
int type_size_min_
The minimum number of arguments an option should be expecting.
Definition Option.hpp:286
CLI11_NODISCARD results_t reduced_results() const
Get a copy of the results.
Definition Option_inl.hpp:444
CLI11_NODISCARD std::string get_type_name() const
Get the full typename for this option.
Definition Option_inl.hpp:505
CLI11_NODISCARD bool check_fname(std::string name) const
Requires "--" to be removed from string.
Definition Option.hpp:646
std::string pname_
A positional name.
Definition Option.hpp:252
int expected_min_
The minimum number of expected values.
Definition Option.hpp:289
Option * ignore_case(bool value=true)
Definition Option_inl.hpp:179
std::set< Option * > needs_
A list of options that are required with this option.
Definition Option.hpp:297
void run_callback()
Process the callback.
Definition Option_inl.hpp:286
CLI11_NODISCARD bool check_sname(std::string name) const
Requires "-" to be removed from string.
Definition Option.hpp:636
bool flag_like_
Specify that the option should act like a flag vs regular option.
Definition Option.hpp:332
CLI11_NODISCARD std::string get_name(bool positional=false, bool all_options=false) const
Gets a comma separated list of names. Will include / prefer the positional name if positional is true...
Definition Option_inl.hpp:233
std::set< Option * > excludes_
A list of options that are excluded with this option.
Definition Option.hpp:300
bool force_callback_
flag indicating that the option should force the callback regardless if any results present
Definition Option.hpp:340
std::vector< std::string > fnames_
a list of flag names with specified default values;
Definition Option.hpp:249
CLI11_NODISCARD int get_items_expected_min() const
The total min number of expected string values to be used.
Definition Option.hpp:572
CLI11_NODISCARD const std::string & matching_name(const Option &other) const
If options share any of the same names, find it.
Definition Option_inl.hpp:312
CLI11_NODISCARD bool check_lname(std::string name) const
Requires "--" to be removed from string.
Definition Option.hpp:641
CLI11_NODISCARD int get_items_expected_max() const
Get the maximum number of items expected to be returned and used for the callback.
Definition Option.hpp:575
std::vector< std::string > snames_
A list of the short names (-a) without the leading dashes.
Definition Option.hpp:239
results_t proc_results_
results after reduction
Definition Option.hpp:319
bool remove_excludes(Option *opt)
Remove needs link from an option. Returns true if the option really was in the needs list.
Definition Option_inl.hpp:169
Option * multi_option_policy(MultiOptionPolicy value=MultiOptionPolicy::Throw)
Take the last argument if given multiple times (or another policy)
Definition Option_inl.hpp:220
Option * excludes(Option *opt)
Sets excluded options.
Definition Option_inl.hpp:154
App * parent_
link back up to the parent App for fallthrough
Definition Option.hpp:307
int expected_max_
The maximum number of expected values.
Definition Option.hpp:291
CLI11_NODISCARD std::string get_flag_value(const std::string &name, std::string input_value) const
Definition Option_inl.hpp:378
CLI11_NODISCARD int get_items_expected() const
The total min number of expected string values to be used.
Definition Option.hpp:580
Validator * get_validator(const std::string &Validator_name="")
Get a named Validator.
Definition Option_inl.hpp:124
bool inject_separator_
flag indicating a separator needs to be injected after each argument call
Definition Option.hpp:336
Option * check(Validator validator, const std::string &validator_name="")
Adds a Validator with a built in type name.
Definition Option_inl.hpp:76
callback_t callback_
Options store a callback to do all the work.
Definition Option.hpp:310
CLI11_NODISCARD bool empty() const
True if the option was not passed.
Definition Option.hpp:360
std::string envname_
If given, check the environment for this option.
Definition Option.hpp:255
Option * ignore_underscore(bool value=true)
Definition Option_inl.hpp:199
std::vector< std::pair< std::string, std::string > > default_flag_values_
Definition Option.hpp:246
std::vector< Validator > validators_
A list of Validators to run on each value parsed.
Definition Option.hpp:294
int type_size_max_
Definition Option.hpp:284
bool allow_extra_args_
Specify that extra args beyond type_size_max should be allowed.
Definition Option.hpp:330
Option * transform(Validator Validator, const std::string &Validator_name="")
Adds a transforming Validator with a built in type name.
Definition Option_inl.hpp:92
std::vector< std::string > lnames_
A list of the long names (--long) without the leading dashes.
Definition Option.hpp:242
Option * add_result(std::string s)
Puts a result at the end.
Definition Option_inl.hpp:424
bool remove_needs(Option *opt)
Remove needs link from an option. Returns true if the option really was in the needs list.
Definition Option_inl.hpp:144
Option * each(const std::function< void(std::string)> &func)
Adds a user supplied function to run on each item passed in (communicate though lambda capture)
Definition Option_inl.hpp:114
results_t results_
complete Results of parsing
Definition Option.hpp:317
Thrown when counting a non-existent option.
Definition Error.hpp:351
Some validators that are provided.
Definition Validators.hpp:55
CLI11_NODISCARD std::string get_description() const
Generate type description information for the Validator.
Definition Validators.hpp:108
CLI11_NODISCARD const std::string & get_name() const
Get the name of the Validator.
Definition Validators.hpp:126