Skip to content

Commit 65442ad

Browse files
LostInCompilationpre-commit-ci[bot]phlptp
authored
A better Help formatter (V2) (#866)
_This is the new PR I've mentioned to work on in PR #858_ ## A better Help Formatter _See below for images of the new help page_ Finally, after a lot of planning, understanding CLI11's codebase, testing and coding, the new default Help Formatter is done. There are a lot of changes to make the help page more readable and closer to UNIX standards, see Changelog below for details. One of the highlights is automatic paragraph formatting with correct line wrapping for App and options/flag descriptions as well as the footer. A goal was to provide more flexibility and better readability for the help page while providing full compatibility with Apps using CLI11 (no breaking changes and no changes to Apps required). Also better support for different terminal sizes. Users can now specify three new optional attributes: `right_column_width_`, `description_paragraph_width_` and `footer_paragraph_width_`. See code documentation for more details. The different columns for options/flags now scale with the set `column_width_` value: Single dash flags occupy 33% of the set `column_width_`, double dash flags and options (like REQUIRED) 66%. These new attributes allow for indirectly respecting terminal geometry, footer paragraph formatting has also been added (#355). This PR also implements the issues #353 and #856. The new help page formatting can also be used as an input for man page generation, since it's oriented on the man page style (#413). [help2man](https://www.gnu.org/software/help2man/) can be used to generate man pages from help output (see comment down below for example). I thoroughly tested this code with all possible combinations of flags, options, positionals, subcommands, validators, ... So far everything works great. I hope this PR looks good and meets all requirements. I'm looking forward to the implementation of this PR into CLI11. If you have any questions or suggestions feel free to comment. ### Fixed/implemented issues by this PR - #353 Better options formatting - #856 Space between options - #355 Footer formatting - #413 Man page generation can be achieved using help2man with the new help formatting - #384 (comment) Better help formatting can be marked as complete ### What about the failing tests? Of course the tests expect the old help text format. This is why 6 of the tests are failing. Since it is a bit of work to migrate the tests to the new help format, I first wanted to push out this PR and get confirmation before I'll update all the tests. So please let me know if this PR gets implemented, what changes should be made and then I'll migrate the tests to the new help format, either in this PR or I'll make a new one. ## Changelog: #### There are _no breaking changes_. Every App using CLI11 will work with this new formatter with no changes required. - Added empty lines at beginning and end of help text - Removed double new-line between option groups for consistency. Now all sections have the same number of new-lines - Switched usage and description order - Only show "Usage"-string if no App name is present. This provides better readability - Made categories (Options, Positionals, ...) capital - Changed `ConfigBase::to_config` to correctly process capital "OPTIONS"-group (only affects descriptions of the config file, not a breaking change) - Added a paragraph formatter function `streamOutAsParagraph` to StringTools.hpp - Made "description" a paragraph block with correct, word respecting line wrapping and indentation (using the new paragraph formatter function) - Made the footer a paragraph block with correct, word respecting line wrapping and indentation - Updated documentation for `column_width_` to make it more clear - Added new member: `right_column_width_`, added getter and setter for `right_column_width_` - Added new member: `description_paragraph_width_`, added getter and setter for `description_paragraph_width_` - Added new member: `footer_paragraph_width_`, added getter and setter for `footer_paragraph_width_ ` - Positionals description are now formatted as paragraph with correct, word respecting line wrapping - Options description are now formatted as paragraph with correct, word respecting line wrapping - Short and long options/flags/names are now correctly formatted to always be at the right position (also for subcommand options/flags) - Short and long options/flags/names column widths scale linearly with the `column_width_` attribute to better adapt to different `column_width_` sizes - Merged PR #860 ## What's planned for the future? - I'm thinking of better formatting the options of flags (like REQUIRED, TEXT, INT, ...) and make them also in a seperate column. This way they would also always be at the same position. However I decided against it for this PR, since I wanted them to be as close as possible to the actual flag. With my implementation it is quite easy to add this change in the future. - Subcommands: I'm planning on better formatting the Subcommands. With this PR only the short and long flags/options of subcommands are better formatted (like it is with the main flags, see images down below). - Maybe implement a different way to display expected data type options (TEXT, INT, ...). For example: `--file-name=<TEXT>` for long flags only and if `disable_flag_override_` is false. - Maybe add something like this: #554 --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Philip Top <[email protected]>
1 parent 924e3e8 commit 65442ad

17 files changed

+467
-184
lines changed

.codecov.yml

+8
Original file line numberDiff line numberDiff line change
@@ -5,3 +5,11 @@ ignore:
55
- "docs"
66
- "test_package"
77
- "fuzz"
8+
9+
parsers:
10+
gcov:
11+
branch_detection:
12+
conditional: yes
13+
loop: yes
14+
method: no
15+
macro: no

examples/CMakeLists.txt

+4-4
Original file line numberDiff line numberDiff line change
@@ -75,8 +75,8 @@ set_property(
7575
"Working on count: 2, direct count: 2, opt count: 2" "Some value: 1.2")
7676
# test shows that the help prints out for unnamed subcommands
7777
add_test(NAME subcom_partitioned_help COMMAND subcom_partitioned --help)
78-
set_property(TEST subcom_partitioned_help PROPERTY PASS_REGULAR_EXPRESSION
79-
"-f,--file TEXT REQUIRED" "-d,--double FLOAT")
78+
set_property(TEST subcom_partitioned_help
79+
PROPERTY PASS_REGULAR_EXPRESSION "-f,[ \\t]*--file TEXT REQUIRED" "-d,--double FLOAT")
8080

8181
####################################################
8282
add_cli_exe(config_app config_app.cpp)
@@ -145,8 +145,8 @@ add_cli_exe(validators validators.cpp)
145145
add_test(NAME validators_help COMMAND validators --help)
146146
set_property(
147147
TEST validators_help
148-
PROPERTY PASS_REGULAR_EXPRESSION " -f,--file TEXT:FILE[\\r\\n\\t ]+File name"
149-
" -v,--value INT:INT in [3 - 6][\\r\\n\\t ]+Value in range")
148+
PROPERTY PASS_REGULAR_EXPRESSION " -f,[ \\t]*--file TEXT:FILE[\\r\\n\\t ]+File name"
149+
" -v,[ \\t]*--value INT:INT in [3 - 6][\\r\\n\\t ]+Value in range")
150150
add_test(NAME validators_file COMMAND validators --file nonex.xxx)
151151
set_property(
152152
TEST validators_file PROPERTY PASS_REGULAR_EXPRESSION "--file: File does not exist: nonex.xxx"

fuzz/fuzzApp.cpp

+1
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ std::optional<std::string>> tcomplex; std::string_view vstrv;
4444
std::shared_ptr<CLI::App> FuzzApp::generateApp() {
4545
auto fApp = std::make_shared<CLI::App>("fuzzing App", "fuzzer");
4646
fApp->set_config("--config");
47+
fApp->set_help_all_flag("--help-all");
4748
fApp->add_flag("-a,--flag");
4849
fApp->add_flag("-b,--flag2,!--nflag2", flag1);
4950
fApp->add_flag("-c{34},--flag3{1}", flagCnt)->disable_flag_override();

include/CLI/App.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -275,7 +275,7 @@ class App {
275275
App *parent_{nullptr};
276276

277277
/// The group membership INHERITABLE
278-
std::string group_{"Subcommands"};
278+
std::string group_{"SUBCOMMANDS"};
279279

280280
/// Alias names for the subcommand
281281
std::vector<std::string> aliases_{};

include/CLI/FormatterFwd.hpp

+33-11
Original file line numberDiff line numberDiff line change
@@ -44,9 +44,18 @@ class FormatterBase {
4444
/// @name Options
4545
///@{
4646

47-
/// The width of the first column
47+
/// The width of the left column (options/flags/subcommands)
4848
std::size_t column_width_{30};
4949

50+
/// The width of the right column (description of options/flags/subcommands)
51+
std::size_t right_column_width_{65};
52+
53+
/// The width of the description paragraph at the top of help
54+
std::size_t description_paragraph_width_{80};
55+
56+
/// The width of the footer paragraph
57+
std::size_t footer_paragraph_width_{80};
58+
5059
/// @brief The required help printout labels (user changeable)
5160
/// Values are Needs, Excludes, etc.
5261
std::map<std::string, std::string> labels_{};
@@ -75,9 +84,18 @@ class FormatterBase {
7584
/// Set the "REQUIRED" label
7685
void label(std::string key, std::string val) { labels_[key] = val; }
7786

78-
/// Set the column width
87+
/// Set the left column width (options/flags/subcommands)
7988
void column_width(std::size_t val) { column_width_ = val; }
8089

90+
/// Set the right column width (description of options/flags/subcommands)
91+
void right_column_width(std::size_t val) { right_column_width_ = val; }
92+
93+
/// Set the description paragraph width at the top of help
94+
void description_paragraph_width(std::size_t val) { description_paragraph_width_ = val; }
95+
96+
/// Set the footer paragraph width
97+
void footer_paragraph_width(std::size_t val) { footer_paragraph_width_ = val; }
98+
8199
///@}
82100
/// @name Getters
83101
///@{
@@ -89,9 +107,18 @@ class FormatterBase {
89107
return labels_.at(key);
90108
}
91109

92-
/// Get the current column width
110+
/// Get the current left column width (options/flags/subcommands)
93111
CLI11_NODISCARD std::size_t get_column_width() const { return column_width_; }
94112

113+
/// Get the current right column width (description of options/flags/subcommands)
114+
CLI11_NODISCARD std::size_t get_right_column_width() const { return right_column_width_; }
115+
116+
/// Get the current description paragraph width at the top of help
117+
CLI11_NODISCARD std::size_t get_description_paragraph_width() const { return description_paragraph_width_; }
118+
119+
/// Get the current footer paragraph width
120+
CLI11_NODISCARD std::size_t get_footer_paragraph_width() const { return footer_paragraph_width_; }
121+
95122
///@}
96123
};
97124

@@ -146,7 +173,7 @@ class Formatter : public FormatterBase {
146173
virtual std::string make_subcommand(const App *sub) const;
147174

148175
/// This prints out a subcommand in help-all
149-
virtual std::string make_expanded(const App *sub) const;
176+
virtual std::string make_expanded(const App *sub, AppFormatMode mode) const;
150177

151178
/// This prints out all the groups of options
152179
virtual std::string make_footer(const App *app) const;
@@ -158,19 +185,14 @@ class Formatter : public FormatterBase {
158185
virtual std::string make_usage(const App *app, std::string name) const;
159186

160187
/// This puts everything together
161-
std::string make_help(const App * /*app*/, std::string, AppFormatMode) const override;
188+
std::string make_help(const App *app, std::string, AppFormatMode mode) const override;
162189

163190
///@}
164191
/// @name Options
165192
///@{
166193

167194
/// This prints out an option help line, either positional or optional form
168-
virtual std::string make_option(const Option *opt, bool is_positional) const {
169-
std::stringstream out;
170-
detail::format_help(
171-
out, make_option_name(opt, is_positional) + make_option_opts(opt), make_option_desc(opt), column_width_);
172-
return out.str();
173-
}
195+
virtual std::string make_option(const Option *, bool) const;
174196

175197
/// @brief This is the name part of an option, Default: left column
176198
virtual std::string make_option_name(const Option *, bool) const;

include/CLI/Option.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ template <typename CRTP> class OptionBase {
5454

5555
protected:
5656
/// The group membership
57-
std::string group_ = std::string("Options");
57+
std::string group_ = std::string("OPTIONS");
5858

5959
/// True if this is a required option
6060
bool required_{false};

include/CLI/StringTools.hpp

+8-3
Original file line numberDiff line numberDiff line change
@@ -141,9 +141,6 @@ inline std::string trim_copy(const std::string &str, const std::string &filter)
141141
std::string s = str;
142142
return trim(s, filter);
143143
}
144-
/// Print a two part "help" string
145-
CLI11_INLINE std::ostream &
146-
format_help(std::ostream &out, std::string name, const std::string &description, std::size_t wid);
147144

148145
/// Print subcommand aliases
149146
CLI11_INLINE std::ostream &format_aliases(std::ostream &out, const std::vector<std::string> &aliases, std::size_t wid);
@@ -263,6 +260,14 @@ CLI11_INLINE std::string extract_binary_string(const std::string &escaped_string
263260
/// process a quoted string, remove the quotes and if appropriate handle escaped characters
264261
CLI11_INLINE bool process_quoted_string(std::string &str, char string_char = '\"', char literal_char = '\'');
265262

263+
/// This function formats the given text as a paragraph with fixed width and applies correct line wrapping
264+
/// with a custom line prefix. The paragraph will get streamed to the given ostrean.
265+
CLI11_INLINE std::ostream &streamOutAsParagraph(std::ostream &out,
266+
const std::string &text,
267+
std::size_t paragraphWidth,
268+
const std::string &linePrefix = "",
269+
bool skipPrefixOnFirstLine = false);
270+
266271
} // namespace detail
267272

268273
// [CLI11:string_tools_hpp:end]

include/CLI/TypeTools.hpp

+12-4
Original file line numberDiff line numberDiff line change
@@ -904,7 +904,7 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
904904
nstring.erase(std::remove(nstring.begin(), nstring.end(), '\''), nstring.end());
905905
return integral_conversion(nstring, output);
906906
}
907-
if(input.compare(0, 2, "0o") == 0) {
907+
if(input.compare(0, 2, "0o") == 0 || input.compare(0, 2, "0O") == 0) {
908908
val = nullptr;
909909
errno = 0;
910910
output_ll = std::strtoull(input.c_str() + 2, &val, 8);
@@ -914,7 +914,10 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
914914
output = static_cast<T>(output_ll);
915915
return (val == (input.c_str() + input.size()) && static_cast<std::uint64_t>(output) == output_ll);
916916
}
917-
if(input.compare(0, 2, "0b") == 0) {
917+
if(input.compare(0, 2, "0b") == 0 || input.compare(0, 2, "0B") == 0) {
918+
// LCOV_EXCL_START
919+
// In some new compilers including the coverage testing one binary strings are handled properly in strtoull
920+
// automatically so this coverage is missing but is well tested in other compilers
918921
val = nullptr;
919922
errno = 0;
920923
output_ll = std::strtoull(input.c_str() + 2, &val, 2);
@@ -923,6 +926,7 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
923926
}
924927
output = static_cast<T>(output_ll);
925928
return (val == (input.c_str() + input.size()) && static_cast<std::uint64_t>(output) == output_ll);
929+
// LCOV_EXCL_STOP
926930
}
927931
return false;
928932
}
@@ -955,7 +959,7 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
955959
nstring.erase(std::remove(nstring.begin(), nstring.end(), '\''), nstring.end());
956960
return integral_conversion(nstring, output);
957961
}
958-
if(input.compare(0, 2, "0o") == 0) {
962+
if(input.compare(0, 2, "0o") == 0 || input.compare(0, 2, "0O") == 0) {
959963
val = nullptr;
960964
errno = 0;
961965
output_ll = std::strtoll(input.c_str() + 2, &val, 8);
@@ -965,7 +969,10 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
965969
output = static_cast<T>(output_ll);
966970
return (val == (input.c_str() + input.size()) && static_cast<std::int64_t>(output) == output_ll);
967971
}
968-
if(input.compare(0, 2, "0b") == 0) {
972+
if(input.compare(0, 2, "0b") == 0 || input.compare(0, 2, "0B") == 0) {
973+
// LCOV_EXCL_START
974+
// In some new compilers including the coverage testing one binary strings are handled properly in strtoll
975+
// automatically so this coverage is missing but is well tested in other compilers
969976
val = nullptr;
970977
errno = 0;
971978
output_ll = std::strtoll(input.c_str() + 2, &val, 2);
@@ -974,6 +981,7 @@ bool integral_conversion(const std::string &input, T &output) noexcept {
974981
}
975982
output = static_cast<T>(output_ll);
976983
return (val == (input.c_str() + input.size()) && static_cast<std::int64_t>(output) == output_ll);
984+
// LCOV_EXCL_STOP
977985
}
978986
return false;
979987
}

include/CLI/impl/App_inl.hpp

+6
Original file line numberDiff line numberDiff line change
@@ -2265,11 +2265,14 @@ CLI11_INLINE void retire_option(App *app, Option *opt) {
22652265
->expected(option_copy->get_expected_min(), option_copy->get_expected_max())
22662266
->allow_extra_args(option_copy->get_allow_extra_args());
22672267

2268+
// LCOV_EXCL_START
2269+
// something odd with coverage on new compilers
22682270
Validator retired_warning{[opt2](std::string &) {
22692271
std::cout << "WARNING " << opt2->get_name() << " is retired and has no effect\n";
22702272
return std::string();
22712273
},
22722274
""};
2275+
// LCOV_EXCL_STOP
22732276
retired_warning.application_index(0);
22742277
opt2->check(retired_warning);
22752278
}
@@ -2287,11 +2290,14 @@ CLI11_INLINE void retire_option(App *app, const std::string &option_name) {
22872290
->type_name("RETIRED")
22882291
->expected(0, 1)
22892292
->default_str("RETIRED");
2293+
// LCOV_EXCL_START
2294+
// something odd with coverage on new compilers
22902295
Validator retired_warning{[opt2](std::string &) {
22912296
std::cout << "WARNING " << opt2->get_name() << " is retired and has no effect\n";
22922297
return std::string();
22932298
},
22942299
""};
2300+
// LCOV_EXCL_STOP
22952301
retired_warning.application_index(0);
22962302
opt2->check(retired_warning);
22972303
}

include/CLI/impl/Config_inl.hpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -517,26 +517,26 @@ ConfigBase::to_config(const App *app, bool default_also, bool write_description,
517517

518518
std::vector<std::string> groups = app->get_groups();
519519
bool defaultUsed = false;
520-
groups.insert(groups.begin(), std::string("Options"));
520+
groups.insert(groups.begin(), std::string("OPTIONS"));
521521
if(write_description && (app->get_configurable() || app->get_parent() == nullptr || app->get_name().empty())) {
522522
out << commentLead << detail::fix_newlines(commentLead, app->get_description()) << '\n';
523523
}
524524
for(auto &group : groups) {
525-
if(group == "Options" || group.empty()) {
525+
if(group == "OPTIONS" || group.empty()) {
526526
if(defaultUsed) {
527527
continue;
528528
}
529529
defaultUsed = true;
530530
}
531-
if(write_description && group != "Options" && !group.empty()) {
531+
if(write_description && group != "OPTIONS" && !group.empty()) {
532532
out << '\n' << commentLead << group << " Options\n";
533533
}
534534
for(const Option *opt : app->get_options({})) {
535535

536536
// Only process options that are configurable
537537
if(opt->get_configurable()) {
538538
if(opt->get_group() != group) {
539-
if(!(group == "Options" && opt->get_group().empty())) {
539+
if(!(group == "OPTIONS" && opt->get_group().empty())) {
540540
continue;
541541
}
542542
}

0 commit comments

Comments
 (0)