mirror of
https://github.com/clap-rs/clap
synced 2024-12-14 06:42:33 +00:00
Use a template to produce the default help message.
This makes some changes to the template system: - Template tags for optional items (like {author}) now expand to nothing when the value is unset instead of a default string (like "unknown author"). - Many template tags now emit line-wrapped output to match write_default_help. - Items with long variants now expand to the appropriate thing for -h vs --help. - The now-obsolete {long-about} tag has been removed. - A few new tags have been added. These are externally-visible changes, but if this makes it into 3.0 that's probably reasonable? Note that line-wrapping can have some odd edge cases since it does not account for preceding/trailing characters on the same line as the tag. This is already the case in master, but will affect some additional tags with this changeset. See #2065 for details. Closes #2002.
This commit is contained in:
parent
ca6c84fa8a
commit
a87320ae88
3 changed files with 107 additions and 154 deletions
|
@ -724,26 +724,30 @@ impl<'help> App<'help> {
|
||||||
/// **NOTE:** The template system is by design very simple. Therefore, the
|
/// **NOTE:** The template system is by design very simple. Therefore, the
|
||||||
/// tags have to be written in the lowercase and without spacing.
|
/// tags have to be written in the lowercase and without spacing.
|
||||||
///
|
///
|
||||||
/// Tags arg given inside curly brackets.
|
/// Tags are given inside curly brackets.
|
||||||
///
|
///
|
||||||
/// Valid tags are:
|
/// Valid tags are:
|
||||||
///
|
///
|
||||||
/// * `{bin}` - Binary name.
|
/// * `{bin}` - Binary name.
|
||||||
/// * `{version}` - Version number.
|
/// * `{version}` - Version number.
|
||||||
/// * `{author}` - Author information.
|
/// * `{author}` - Author information.
|
||||||
/// * `{about}` - General description (from [`App::about`])
|
/// * `{author-with-newline}` - Author followed by `\n`.
|
||||||
/// * `{usage}` - Automatically generated or given usage string.
|
/// * `{about}` - Description of the program.
|
||||||
/// * `{all-args}` - Help for all arguments (options, flags, positionals arguments,
|
/// * `{about-with-newline}` - Description followed by `\n`.
|
||||||
/// and subcommands) including titles.
|
/// * `{usage}` - Automatically generated or given usage string.
|
||||||
/// * `{unified}` - Unified help for options and flags. Note, you must *also* set
|
/// * `{all-args}` - Help for all arguments (options, flags, positional
|
||||||
/// [`AppSettings::UnifiedHelpMessage`] to fully merge both options and
|
/// arguments, and subcommands) including titles.
|
||||||
/// flags, otherwise the ordering is "best effort"
|
/// * `{unified}` - Unified help for options and flags. Note, you must *also*
|
||||||
/// * `{flags}` - Help for flags.
|
/// set [`AppSettings::UnifiedHelpMessage`] to fully merge both
|
||||||
/// * `{options}` - Help for options.
|
/// options and flags, otherwise the ordering is "best effort".
|
||||||
/// * `{positionals}` - Help for positionals arguments.
|
/// * `{flags}` - Help for flags.
|
||||||
/// * `{subcommands}` - Help for subcommands.
|
/// * `{options}` - Help for options.
|
||||||
/// * `{after-help}` - Help from [`App::after_help`]
|
/// * `{positionals}` - Help for positional arguments.
|
||||||
/// * `{before-help}` - Help from [`App::before_help`]
|
/// * `{subcommands}` - Help for subcommands.
|
||||||
|
/// * `{after-help}` - Info to be displayed after the help message.
|
||||||
|
/// * `{after-help-padded}` - After help message with additional spacing.
|
||||||
|
/// * `{before-help}` - Info to be displayed before the help message.
|
||||||
|
/// * `{before-help-padded}` - Before help message with additional spacing.
|
||||||
///
|
///
|
||||||
/// # Examples
|
/// # Examples
|
||||||
///
|
///
|
||||||
|
@ -754,9 +758,6 @@ impl<'help> App<'help> {
|
||||||
/// .help_template("{bin} ({version}) - {usage}")
|
/// .help_template("{bin} ({version}) - {usage}")
|
||||||
/// # ;
|
/// # ;
|
||||||
/// ```
|
/// ```
|
||||||
/// [`App::about`]: ./struct.App.html#method.about
|
|
||||||
/// [`App::after_help`]: ./struct.App.html#method.after_help
|
|
||||||
/// [`App::before_help`]: ./struct.App.html#method.before_help
|
|
||||||
/// [`AppSettings::UnifiedHelpMessage`]: ./enum.AppSettings.html#variant.UnifiedHelpMessage
|
/// [`AppSettings::UnifiedHelpMessage`]: ./enum.AppSettings.html#variant.UnifiedHelpMessage
|
||||||
pub fn help_template<S: Into<&'help str>>(mut self, s: S) -> Self {
|
pub fn help_template<S: Into<&'help str>>(mut self, s: S) -> Self {
|
||||||
self.template = Some(s.into());
|
self.template = Some(s.into());
|
||||||
|
@ -2448,11 +2449,6 @@ impl<'help> App<'help> {
|
||||||
!self.args.is_empty()
|
!self.args.is_empty()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(crate) fn has_opts(&self) -> bool {
|
|
||||||
self.get_opts_no_heading().count() > 0
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub(crate) fn has_flags(&self) -> bool {
|
pub(crate) fn has_flags(&self) -> bool {
|
||||||
self.get_flags_no_heading().count() > 0
|
self.get_flags_no_heading().count() > 0
|
||||||
|
|
|
@ -74,6 +74,14 @@ pub(crate) struct Help<'help, 'app, 'parser, 'writer> {
|
||||||
|
|
||||||
// Public Functions
|
// Public Functions
|
||||||
impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
|
const DEFAULT_TEMPLATE: &'static str = "\
|
||||||
|
{before-help-padded}{bin} {version}\n\
|
||||||
|
{author-with-newline}{about-with-newline}\n\
|
||||||
|
USAGE:\n {usage}\n\
|
||||||
|
\n\
|
||||||
|
{all-args}{after-help-padded}\
|
||||||
|
";
|
||||||
|
|
||||||
/// Create a new `Help` instance.
|
/// Create a new `Help` instance.
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
w: HelpWriter<'writer>,
|
w: HelpWriter<'writer>,
|
||||||
|
@ -116,7 +124,7 @@ impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
} else if let Some(tmpl) = self.parser.app.template {
|
} else if let Some(tmpl) = self.parser.app.template {
|
||||||
self.write_templated_help(tmpl)?;
|
self.write_templated_help(tmpl)?;
|
||||||
} else {
|
} else {
|
||||||
self.write_default_help()?;
|
self.write_templated_help(Self::DEFAULT_TEMPLATE)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.none("\n")?;
|
self.none("\n")?;
|
||||||
|
@ -811,13 +819,6 @@ impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Writes version of a Parser Object to the wrapped stream.
|
|
||||||
fn write_version(&mut self) -> io::Result<()> {
|
|
||||||
debug!("Help::write_version");
|
|
||||||
self.none(self.parser.app.version.unwrap_or(""))?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Writes binary name of a Parser Object to the wrapped stream.
|
/// Writes binary name of a Parser Object to the wrapped stream.
|
||||||
fn write_bin_name(&mut self) -> io::Result<()> {
|
fn write_bin_name(&mut self) -> io::Result<()> {
|
||||||
debug!("Help::write_bin_name");
|
debug!("Help::write_bin_name");
|
||||||
|
@ -839,98 +840,6 @@ impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Writes default help for a Parser Object to the wrapped stream.
|
|
||||||
pub(crate) fn write_default_help(&mut self) -> ClapResult<()> {
|
|
||||||
debug!("Help::write_default_help");
|
|
||||||
if self.use_long {
|
|
||||||
if let Some(h) = self
|
|
||||||
.parser
|
|
||||||
.app
|
|
||||||
.before_long_help
|
|
||||||
.or_else(|| self.parser.app.before_help)
|
|
||||||
{
|
|
||||||
self.write_before_after_help(h)?;
|
|
||||||
self.none("\n\n")?;
|
|
||||||
}
|
|
||||||
} else if let Some(h) = self
|
|
||||||
.parser
|
|
||||||
.app
|
|
||||||
.before_help
|
|
||||||
.or_else(|| self.parser.app.before_long_help)
|
|
||||||
{
|
|
||||||
self.write_before_after_help(h)?;
|
|
||||||
self.none("\n\n")?;
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! write_thing {
|
|
||||||
($thing:expr) => {{
|
|
||||||
self.none(&wrap_help(&$thing, self.term_w))?;
|
|
||||||
self.none("\n")?
|
|
||||||
}};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Print the version
|
|
||||||
self.write_bin_name()?;
|
|
||||||
self.none(" ")?;
|
|
||||||
self.write_version()?;
|
|
||||||
self.none("\n")?;
|
|
||||||
|
|
||||||
if let Some(author) = self.parser.app.author {
|
|
||||||
write_thing!(author);
|
|
||||||
}
|
|
||||||
|
|
||||||
if self.use_long && self.parser.app.long_about.is_some() {
|
|
||||||
debug!("Help::write_default_help: writing long about");
|
|
||||||
write_thing!(self.parser.app.long_about.unwrap());
|
|
||||||
} else if self.parser.app.about.is_some() {
|
|
||||||
debug!("Help::write_default_help: writing about");
|
|
||||||
write_thing!(self.parser.app.about.unwrap());
|
|
||||||
}
|
|
||||||
|
|
||||||
self.none("\n")?;
|
|
||||||
self.warning("USAGE:")?;
|
|
||||||
self.none(&format!(
|
|
||||||
"\n{}{}\n\n",
|
|
||||||
TAB,
|
|
||||||
Usage::new(self.parser).create_usage_no_title(&[])
|
|
||||||
))?;
|
|
||||||
|
|
||||||
let flags = self.parser.has_flags();
|
|
||||||
let pos = self.parser.has_positionals();
|
|
||||||
let opts = self.parser.has_opts();
|
|
||||||
let subcmds = self.parser.has_subcommands();
|
|
||||||
|
|
||||||
if flags || opts || pos || subcmds {
|
|
||||||
self.write_all_args()?;
|
|
||||||
}
|
|
||||||
|
|
||||||
if self.use_long {
|
|
||||||
if let Some(h) = self
|
|
||||||
.parser
|
|
||||||
.app
|
|
||||||
.after_long_help
|
|
||||||
.or_else(|| self.parser.app.after_help)
|
|
||||||
{
|
|
||||||
if flags || opts || pos || subcmds {
|
|
||||||
self.none("\n\n")?;
|
|
||||||
}
|
|
||||||
self.write_before_after_help(h)?;
|
|
||||||
}
|
|
||||||
} else if let Some(h) = self
|
|
||||||
.parser
|
|
||||||
.app
|
|
||||||
.after_help
|
|
||||||
.or_else(|| self.parser.app.after_long_help)
|
|
||||||
{
|
|
||||||
if flags || opts || pos || subcmds {
|
|
||||||
self.none("\n\n")?;
|
|
||||||
}
|
|
||||||
self.write_before_after_help(h)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
self.writer.flush().map_err(Error::from)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Possible results for a copying function that stops when a given
|
/// Possible results for a copying function that stops when a given
|
||||||
|
@ -1034,28 +943,13 @@ fn copy_and_capture<R: Read, W: Write>(
|
||||||
impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
/// Write help to stream for the parser in the format defined by the template.
|
/// Write help to stream for the parser in the format defined by the template.
|
||||||
///
|
///
|
||||||
/// Tags arg given inside curly brackets:
|
/// For details about the template language see [`App::help_template`].
|
||||||
/// Valid tags are:
|
|
||||||
/// * `{bin}` - Binary name.
|
|
||||||
/// * `{version}` - Version number.
|
|
||||||
/// * `{author}` - Author information.
|
|
||||||
/// * `{usage}` - Automatically generated or given usage string.
|
|
||||||
/// * `{all-args}` - Help for all arguments (options, flags, positionals arguments,
|
|
||||||
/// and subcommands) including titles.
|
|
||||||
/// * `{unified}` - Unified help for options and flags.
|
|
||||||
/// * `{flags}` - Help for flags.
|
|
||||||
/// * `{options}` - Help for options.
|
|
||||||
/// * `{positionals}` - Help for positionals arguments.
|
|
||||||
/// * `{subcommands}` - Help for subcommands.
|
|
||||||
/// * `{after-help}` - Info to be displayed after the help message.
|
|
||||||
/// * `{before-help}` - Info to be displayed before the help message.
|
|
||||||
///
|
///
|
||||||
/// The template system is, on purpose, very simple. Therefore, the tags have to be written
|
/// [`App::help_template`]: ./struct.App.html#method.help_template
|
||||||
/// in the lowercase and without spacing.
|
|
||||||
fn write_templated_help(&mut self, template: &str) -> ClapResult<()> {
|
fn write_templated_help(&mut self, template: &str) -> ClapResult<()> {
|
||||||
debug!("Help::write_templated_help");
|
debug!("Help::write_templated_help");
|
||||||
let mut tmplr = Cursor::new(&template);
|
let mut tmplr = Cursor::new(&template);
|
||||||
let mut tag_buf = Cursor::new(vec![0u8; 15]);
|
let mut tag_buf = Cursor::new(vec![0u8; 20]);
|
||||||
|
|
||||||
// The strategy is to copy the template from the reader to wrapped stream
|
// The strategy is to copy the template from the reader to wrapped stream
|
||||||
// until a tag is found. Depending on its value, the appropriate content is copied
|
// until a tag is found. Depending on its value, the appropriate content is copied
|
||||||
|
@ -1083,16 +977,41 @@ impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
self.write_bin_name()?;
|
self.write_bin_name()?;
|
||||||
}
|
}
|
||||||
b"version" => {
|
b"version" => {
|
||||||
self.none(self.parser.app.version.unwrap_or("unknown version"))?;
|
if let Some(output) = self.parser.app.version {
|
||||||
|
self.none(output)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b"author" => {
|
b"author" => {
|
||||||
self.none(self.parser.app.author.unwrap_or("unknown author"))?;
|
if let Some(output) = self.parser.app.author {
|
||||||
|
self.none(output)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b"author-with-newline" => {
|
||||||
|
if let Some(output) = self.parser.app.author {
|
||||||
|
self.none(&wrap_help(output, self.term_w))?;
|
||||||
|
self.none("\n")?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b"about" => {
|
b"about" => {
|
||||||
self.none(self.parser.app.about.unwrap_or("unknown about"))?;
|
let about = if self.use_long {
|
||||||
|
self.parser.app.long_about.or(self.parser.app.about)
|
||||||
|
} else {
|
||||||
|
self.parser.app.about
|
||||||
|
};
|
||||||
|
if let Some(output) = about {
|
||||||
|
self.none(output)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b"long-about" => {
|
b"about-with-newline" => {
|
||||||
self.none(self.parser.app.long_about.unwrap_or("unknown about"))?;
|
let about = if self.use_long {
|
||||||
|
self.parser.app.long_about.or(self.parser.app.about)
|
||||||
|
} else {
|
||||||
|
self.parser.app.about
|
||||||
|
};
|
||||||
|
if let Some(output) = about {
|
||||||
|
self.none(&wrap_help(output, self.term_w))?;
|
||||||
|
self.none("\n")?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b"usage" => {
|
b"usage" => {
|
||||||
self.none(&Usage::new(self.parser).create_usage_no_title(&[]))?;
|
self.none(&Usage::new(self.parser).create_usage_no_title(&[]))?;
|
||||||
|
@ -1124,10 +1043,52 @@ impl<'help, 'app, 'parser, 'writer> Help<'help, 'app, 'parser, 'writer> {
|
||||||
self.write_subcommands(self.parser.app)?;
|
self.write_subcommands(self.parser.app)?;
|
||||||
}
|
}
|
||||||
b"after-help" => {
|
b"after-help" => {
|
||||||
self.none(self.parser.app.after_help.unwrap_or("unknown after-help"))?;
|
let after_help = if self.use_long {
|
||||||
|
self.parser
|
||||||
|
.app
|
||||||
|
.after_long_help
|
||||||
|
.or(self.parser.app.after_help)
|
||||||
|
} else {
|
||||||
|
self.parser.app.after_help
|
||||||
|
};
|
||||||
|
if let Some(output) = after_help {
|
||||||
|
self.write_before_after_help(output)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b"after-help-padded" => {
|
||||||
|
let after_help = if self.use_long {
|
||||||
|
self.parser
|
||||||
|
.app
|
||||||
|
.after_long_help
|
||||||
|
.or(self.parser.app.after_help)
|
||||||
|
} else {
|
||||||
|
self.parser.app.after_help
|
||||||
|
};
|
||||||
|
if let Some(output) = after_help {
|
||||||
|
self.none("\n\n")?;
|
||||||
|
self.write_before_after_help(output)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
b"before-help" => {
|
b"before-help" => {
|
||||||
self.none(self.parser.app.before_help.unwrap_or("unknown before-help"))?;
|
let before_help = if self.use_long {
|
||||||
|
self.parser.app.before_long_help
|
||||||
|
} else {
|
||||||
|
self.parser.app.before_help
|
||||||
|
};
|
||||||
|
if let Some(output) = before_help {
|
||||||
|
self.write_before_after_help(output)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
b"before-help-padded" => {
|
||||||
|
let before_help = if self.use_long {
|
||||||
|
self.parser.app.before_long_help
|
||||||
|
} else {
|
||||||
|
self.parser.app.before_help
|
||||||
|
};
|
||||||
|
if let Some(output) = before_help {
|
||||||
|
self.write_before_after_help(output)?;
|
||||||
|
self.none("\n\n")?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// Unknown tag, write it back.
|
// Unknown tag, write it back.
|
||||||
r => {
|
r => {
|
||||||
|
|
|
@ -1790,10 +1790,6 @@ impl<'help, 'app> Parser<'help, 'app> {
|
||||||
self.app.has_args()
|
self.app.has_args()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn has_opts(&self) -> bool {
|
|
||||||
self.app.has_opts()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn has_flags(&self) -> bool {
|
pub(crate) fn has_flags(&self) -> bool {
|
||||||
self.app.has_flags()
|
self.app.has_flags()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue