Lint CLI help texts.

This commit is contained in:
Justus Winter 2024-12-08 14:58:36 +01:00
parent edc803eb21
commit 9b00fa43da
No known key found for this signature in database
GPG Key ID: 686F55B4AB2B3386

107
build.rs
View File

@ -26,6 +26,7 @@ fn main() {
let mut sq = cli::build(false);
generate_shell_completions(&mut sq).unwrap();
generate_man_pages(&sq).unwrap();
lint_help_texts(&sq).unwrap();
}
/// Variable name to control the asset out directory with.
@ -72,3 +73,109 @@ fn generate_man_pages(sq: &clap::Command) -> Result<()> {
Ok(())
}
/// Lints the help texts.
fn lint_help_texts(sq: &clap::Command) -> Result<()> {
let mut lints = Vec::new();
walk(&mut Vec::new(), sq,
&mut |path: &[&str], c: &clap::Command| {
let top_level = path.len() == 1;
let path = path.join(" ");
lint_short_long(&mut lints, &path,
c.get_about(), c.get_long_about());
for arg in c.get_arguments()
.filter(|a| a.get_id().as_str() != "help")
.filter(|a| ! a.is_global_set() || top_level)
{
let slug = format!("{} {}", path,
if let Some(l) = arg.get_long() {
format!("--{}", l)
} else if let Some(s) = arg.get_short() {
format!("-{}", s)
} else {
arg.get_id().as_str().to_string()
});
lint_short_long(&mut lints, &slug,
arg.get_help(), arg.get_long_help());
}
Ok(())
},
&mut |_, _, _| Ok(()))?;
if lints.is_empty() {
Ok(())
} else {
println!("cargo:warning=Linting help texts found {} issues",
lints.len());
println!("cargo:warning=");
for lint in lints {
println!("cargo:warning=lint: {}", lint);
}
Err(anyhow::anyhow!("linting help texts failed"))
}
}
use clap::builder::StyledStr;
/// Lints short and long about and help texts.
fn lint_short_long(lints: &mut Vec<String>,
slug: &str,
short: Option<&StyledStr>,
long: Option<&StyledStr>)
{
let short = if let Some(short) = short {
short.to_string()
} else {
lints.push(format!("{}: no short help", slug));
return;
};
if short.contains(". ") {
lints.push(format!("{}: short contains more than one sentence", slug));
}
if short.ends_with(".") {
lints.push(format!("{}: short ends in period", slug));
}
let long = if let Some(long) = long {
long.to_string()
} else {
return;
};
if ! long.starts_with(&format!("{}\n\n", short)) {
lints.push(format!("{}: long help doesn't start with subject (short help + \\n\\n)", slug));
}
if long.split("\n").count() == 1 {
lints.push(format!("{}: long help consists of a single line", slug));
}
}
fn walk<'sq, F, G, R>(path: &mut Vec<&'sq str>, c: &'sq clap::Command,
fun0: &mut F,
fun1: &mut G)
-> Result<R>
where
F: FnMut(&[&str], &clap::Command) -> Result<()>,
G: FnMut(&[&str], &clap::Command, &[R]) -> Result<R>,
{
path.push(c.get_name());
(fun0)(&path, c)?;
let mut r = Vec::new();
for s in c.get_subcommands().filter(|cmd| cmd.get_name() != "help") {
r.push(walk(path, s, fun0, fun1)?);
}
let r = (fun1)(&path, c, &r)?;
path.pop();
Ok(r)
}