Perinci-CmdLine-Help
view release on metacpan or search on metacpan
lib/Perinci/CmdLine/Help.pm view on Meta::CPAN
},
meta_is_normalized => {
schema => 'bool*',
},
common_opts => {
schema => 'hash*',
default => {},
},
per_arg_json => {
schema => 'bool*',
},
per_arg_yaml => {
schema => 'bool*',
},
ggls_res => {
summary => 'Full result from gen_getopt_long_spec_from_meta()',
schema => 'array*', # XXX envres
description => <<'_',
If you already call <pm:Perinci::Sub::GetArgs::Argv>'s
`gen_getopt_long_spec_from_meta()`, you can pass the _full_ enveloped result
here, to avoid calculating twice.
_
},
lang => {
summary => "Will be passed to Perinci::Sub::To::CLIDocData's gen_cli_doc_data_from_meta()",
schema => 'str*',
},
mark_different_lang => {
summary => "Will be passed to Perinci::Sub::To::CLIDocData's gen_cli_doc_data_from_meta()",
schema => 'bool*',
},
},
};
sub gen_help {
no warnings 'once';
require Text::Wrap;
my %args = @_;
local $Text::Wrap::columns = $ENV{COLUMNS} // 80;
my $meta = $args{meta} or return [400, 'Please specify meta'];
unless ($args{meta_is_normalized}) {
require Perinci::Sub::Normalize;
$meta = Perinci::Sub::Normalize::normalize_function_metadata($meta);
}
my $common_opts = $args{common_opts} // {};
my @help;
# summary
my $progname = $args{program_name};
{
my $sum = $args{program_summary} // $meta->{summary};
last unless $sum;
push @help, $progname, " - ", $sum, "\n\n";
}
my $clidocdata;
# usage
push @help, "Usage:\n";
{
for (sort {
($common_opts->{$a}{order} // 99) <=>
($common_opts->{$b}{order} // 99) ||
$a cmp $b
} keys %$common_opts) {
my $co = $common_opts->{$_};
next unless $co->{usage};
push @help, " $progname $co->{usage}\n";
}
require Perinci::Sub::To::CLIDocData;
my $res = Perinci::Sub::To::CLIDocData::gen_cli_doc_data_from_meta(
meta => $meta, meta_is_normalized => 1,
common_opts => $common_opts,
per_arg_json => $args{per_arg_json},
per_arg_yaml => $args{per_arg_yaml},
(ggls_res => $args{ggls_res}) x defined($args{ggls_res}),
(lang => $args{lang}) x defined($args{lang}),
(mark_different_lang => $args{mark_different_lang}) x defined($args{mark_different_lang}),
);
die [500, "gen_cli_doc_data_from_meta failed: ".
"$res->[0] - $res->[1]"] unless $res->[0] == 200;
$clidocdata = $res->[2];
my $usage = $clidocdata->{usage_line};
$usage =~ s/\[\[prog\]\]/$progname/;
local $Text::Wrap::break = '(?=\s)\X|(?<=\\|)';
push @help, Text::Wrap::wrap(" ", " ", "$usage\n");
}
# subcommands
{
my $subcommands = $args{subcommands} or last;
push @help, "\nSubcommands:\n";
if (keys(%$subcommands) >= 12) {
# comma-separated list
push @help, Text::Wrap::wrap(
" ", " ", join(", ", sort keys %$subcommands)), "\n";
} else {
for my $sc_name (sort keys %$subcommands) {
my $sc_spec = $subcommands->{$sc_name};
next unless $sc_spec->{show_in_help} //1;
push @help, " $sc_name\n";
}
}
}
# example
{
# XXX categorize too, like options
last unless @{ $clidocdata->{examples} };
push @help, "\nExamples:\n";
my $i = 0;
my $egs = $clidocdata->{examples};
for my $eg (@$egs) {
$i++;
my $cmdline = $eg->{cmdline};
$cmdline =~ s/\[\[prog\]\]/$progname/;
push @help, "\n" if $eg->{summary} && $i > 1;
if ($eg->{summary}) {
push @help, " $eg->{summary}:\n";
} else {
push @help, "\n";
}
push @help, " % $cmdline\n";
}
}
# description
{
# XXX use proper alt. search
my $desc = $args{program_description} //
$meta->{'description.alt.env.cmdline'} // $meta->{description};
last unless $desc;
$desc =~ s/\A\n+//;
$desc =~ s/\n+\z//;
push @help, "\n", $desc, "\n" if $desc =~ /\S/;
}
# options
{
require Data::Dmp;
my $opts = $clidocdata->{opts};
last unless keys %$opts;
# find all the categories
my %options_by_cat; # val=[options...]
for my $optkey (keys %$opts) {
for my $cat (@{ $opts->{$optkey}{categories} }) {
push @{ $options_by_cat{$cat} }, $optkey;
}
}
my $cats_spec = $clidocdata->{option_categories};
for my $cat (sort {
($cats_spec->{$a}{order} // 50) <=> ($cats_spec->{$b}{order} // 50)
|| $a cmp $b }
keys %options_by_cat) {
# find the longest option
my @opts = sort {length($b)<=>length($a)}
@{ $options_by_cat{$cat} };
my $len = length($opts[0]);
# sort again by name
@opts = sort {
(my $a_without_dash = $a) =~ s/^-+//;
(my $b_without_dash = $b) =~ s/^-+//;
lc($a) cmp lc($b);
} @opts;
push @help, "\n$cat:\n";
for my $opt (@opts) {
my $ospec = $opts->{$opt};
my $arg_spec = $ospec->{arg_spec};
next if grep {$_ eq 'hidden'} @{$arg_spec->{tags} // []};
my $is_bool = $arg_spec->{schema} &&
$arg_spec->{schema}[0] eq 'bool';
my $show_default = exists($ospec->{default}) &&
!$is_bool && !$ospec->{is_base64} &&
!$ospec->{is_json} && !$ospec->{is_yaml} &&
!$ospec->{is_alias};
my $add_sum = '';
if ($ospec->{is_base64}) {
$add_sum = " (as base64-encoded str)";
} elsif ($ospec->{is_json}) {
$add_sum = " (as JSON-encoded str)";
} elsif ($ospec->{is_yaml}) {
$add_sum = " (as YAML-encoded str)";
}
my $argv = '';
if (!$ospec->{main_opt} && defined($ospec->{pos})) {
if ($ospec->{slurpy} // $ospec->{greedy}) {
$argv = " (=arg[$ospec->{pos}-])";
} else {
$argv = " (=arg[$ospec->{pos}])";
}
}
my $cmdline_src = '';
if (!$ospec->{main_opt} && defined($arg_spec->{cmdline_src})) {
$cmdline_src = " (or from $arg_spec->{cmdline_src})";
$cmdline_src =~ s!_or_!/!g;
}
push @help, sprintf(
" %-${len}s %s%s%s%s%s\n",
$opt,
Text::Wrap::wrap("", " " x (2+$len+2 +2),
$ospec->{summary}//''),
$add_sum,
$argv,
$cmdline_src,
($show_default && defined($ospec->{default}) ?
" [".Data::Dmp::dmp($ospec->{default})."]":""),
( run in 0.898 second using v1.01-cache-2.11-cpan-ceb78f64989 )