mirror of
https://github.com/uutils/coreutils
synced 2024-12-14 07:12:44 +00:00
uucore: add filename as argument in help_usage and help_section
uucore: make help_section and help_usage take an argument to select a file
This commit is contained in:
parent
c8e88e1898
commit
a3a69cf919
4 changed files with 176 additions and 59 deletions
|
@ -12,8 +12,8 @@ use uucore::{encoding::Format, error::UResult, help_section, help_usage};
|
||||||
|
|
||||||
pub mod base_common;
|
pub mod base_common;
|
||||||
|
|
||||||
const ABOUT: &str = help_section!("about");
|
const ABOUT: &str = help_section!("about", "base32.md");
|
||||||
const USAGE: &str = help_usage!();
|
const USAGE: &str = help_usage!("base32.md");
|
||||||
|
|
||||||
#[uucore::main]
|
#[uucore::main]
|
||||||
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
|
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
|
||||||
|
|
|
@ -13,8 +13,8 @@ use uucore::{encoding::Format, error::UResult, help_section, help_usage};
|
||||||
|
|
||||||
use std::io::{stdin, Read};
|
use std::io::{stdin, Read};
|
||||||
|
|
||||||
const ABOUT: &str = help_section!("about");
|
const ABOUT: &str = help_section!("about", "base64.md");
|
||||||
const USAGE: &str = help_usage!();
|
const USAGE: &str = help_usage!("base64.md");
|
||||||
|
|
||||||
#[uucore::main]
|
#[uucore::main]
|
||||||
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
|
pub fn uumain(args: impl uucore::Args) -> UResult<()> {
|
||||||
|
|
|
@ -22,9 +22,9 @@ pub mod format;
|
||||||
pub mod options;
|
pub mod options;
|
||||||
mod units;
|
mod units;
|
||||||
|
|
||||||
const ABOUT: &str = help_section!("about");
|
const ABOUT: &str = help_section!("about", "numfmt.md");
|
||||||
const LONG_HELP: &str = help_section!("long help");
|
const LONG_HELP: &str = help_section!("long help", "numfmt.md");
|
||||||
const USAGE: &str = help_usage!();
|
const USAGE: &str = help_usage!("numfmt.md");
|
||||||
|
|
||||||
fn handle_args<'a>(args: impl Iterator<Item = &'a str>, options: &NumfmtOptions) -> UResult<()> {
|
fn handle_args<'a>(args: impl Iterator<Item = &'a str>, options: &NumfmtOptions) -> UResult<()> {
|
||||||
for l in args {
|
for l in args {
|
||||||
|
|
|
@ -37,35 +37,114 @@ pub fn main(_args: TokenStream, stream: TokenStream) -> TokenStream {
|
||||||
TokenStream::from(new)
|
TokenStream::from(new)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_help(section: &str) -> String {
|
/// Get the usage from the "Usage" section in the help file.
|
||||||
|
///
|
||||||
|
/// The usage is assumed to be surrounded by markdown code fences. It may span
|
||||||
|
/// multiple lines. The first word of each line is assumed to be the name of
|
||||||
|
/// the util and is replaced by "{}" so that the output of this function can be
|
||||||
|
/// used with `uucore::format_usage`.
|
||||||
|
#[proc_macro]
|
||||||
|
pub fn help_usage(input: TokenStream) -> TokenStream {
|
||||||
|
let input: Vec<TokenTree> = input.into_iter().collect();
|
||||||
|
let filename = get_argument(&input, 0, "filename");
|
||||||
|
let text: String = parse_usage(&parse_help("usage", &filename));
|
||||||
|
TokenTree::Literal(Literal::string(&text)).into()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Reads a section from a file of the util as a `str` literal.
|
||||||
|
///
|
||||||
|
/// It reads from the file specified as the second argument, relative to the
|
||||||
|
/// crate root. The contents of this file are read verbatim, without parsing or
|
||||||
|
/// escaping. The name of the help file should match the name of the util.
|
||||||
|
/// I.e. numfmt should have a file called `numfmt.md`. By convention, the file
|
||||||
|
/// should start with a top-level section with the name of the util. The other
|
||||||
|
/// sections must start with 2 `#` characters. Capitalization of the sections
|
||||||
|
/// does not matter. Leading and trailing whitespace of each section will be
|
||||||
|
/// removed.
|
||||||
|
///
|
||||||
|
/// Example:
|
||||||
|
/// ```md
|
||||||
|
/// # numfmt
|
||||||
|
/// ## About
|
||||||
|
/// Convert numbers from/to human-readable strings
|
||||||
|
///
|
||||||
|
/// ## Long help
|
||||||
|
/// This text will be the long help
|
||||||
|
/// ```
|
||||||
|
///
|
||||||
|
/// ```rust,ignore
|
||||||
|
/// help_section!("about", "numfmt.md");
|
||||||
|
/// ```
|
||||||
|
#[proc_macro]
|
||||||
|
pub fn help_section(input: TokenStream) -> TokenStream {
|
||||||
|
let input: Vec<TokenTree> = input.into_iter().collect();
|
||||||
|
let section = get_argument(&input, 0, "section");
|
||||||
|
let filename = get_argument(&input, 1, "filename");
|
||||||
|
let text = parse_help(§ion, &filename);
|
||||||
|
TokenTree::Literal(Literal::string(&text)).into()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get an argument from the input vector of `TokenTree`.
|
||||||
|
///
|
||||||
|
/// Asserts that the argument is a string literal and returns the string value,
|
||||||
|
/// otherwise it panics with an error.
|
||||||
|
fn get_argument(input: &[TokenTree], index: usize, name: &str) -> String {
|
||||||
|
// Multiply by two to ignore the `','` in between the arguments
|
||||||
|
let string = match &input.get(index * 2) {
|
||||||
|
Some(TokenTree::Literal(lit)) => lit.to_string(),
|
||||||
|
Some(_) => panic!("Argument {} should be a string literal.", index),
|
||||||
|
None => panic!("Missing argument at index {} for {}", index, name),
|
||||||
|
};
|
||||||
|
|
||||||
|
string
|
||||||
|
.parse::<String>()
|
||||||
|
.unwrap()
|
||||||
|
.strip_prefix('"')
|
||||||
|
.unwrap()
|
||||||
|
.strip_suffix('"')
|
||||||
|
.unwrap()
|
||||||
|
.to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Read the help file and extract a section
|
||||||
|
fn parse_help(section: &str, filename: &str) -> String {
|
||||||
let section = section.to_lowercase();
|
let section = section.to_lowercase();
|
||||||
let section = section.trim_matches('"');
|
let section = section.trim_matches('"');
|
||||||
let mut content = String::new();
|
let mut content = String::new();
|
||||||
let mut path = PathBuf::from(std::env::var("CARGO_MANIFEST_DIR").unwrap());
|
let mut path = PathBuf::from(std::env::var("CARGO_MANIFEST_DIR").unwrap());
|
||||||
|
|
||||||
// The package name will be something like uu_numfmt, hence we split once
|
path.push(filename);
|
||||||
// on '_' and take the second element. The help section should then be in a
|
|
||||||
// file called numfmt.md
|
|
||||||
path.push(format!(
|
|
||||||
"{}.md",
|
|
||||||
std::env::var("CARGO_PKG_NAME")
|
|
||||||
.unwrap()
|
|
||||||
.split_once('_')
|
|
||||||
.unwrap()
|
|
||||||
.1,
|
|
||||||
));
|
|
||||||
|
|
||||||
File::open(path)
|
File::open(path)
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.read_to_string(&mut content)
|
.read_to_string(&mut content)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
|
parse_help_section(section, &content)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Get a single section from content
|
||||||
|
///
|
||||||
|
/// The section must be a second level section (i.e. start with `##`).
|
||||||
|
fn parse_help_section(section: &str, content: &str) -> String {
|
||||||
|
fn is_section_header(line: &str, section: &str) -> bool {
|
||||||
|
line.strip_prefix("##")
|
||||||
|
.map_or(false, |l| l.trim().to_lowercase() == section)
|
||||||
|
}
|
||||||
|
|
||||||
|
// We cannot distinguish between an empty or non-existing section below,
|
||||||
|
// so we do a quick test to check whether the section exists to provide
|
||||||
|
// a nice error message.
|
||||||
|
if content.lines().all(|l| !is_section_header(l, section)) {
|
||||||
|
panic!(
|
||||||
|
"The section '{}' could not be found in the help file. Maybe it is spelled wrong?",
|
||||||
|
section
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
content
|
content
|
||||||
.lines()
|
.lines()
|
||||||
.skip_while(|&l| {
|
.skip_while(|&l| !is_section_header(l, section))
|
||||||
l.strip_prefix("##")
|
|
||||||
.map_or(true, |l| l.trim().to_lowercase() != section)
|
|
||||||
})
|
|
||||||
.skip(1)
|
.skip(1)
|
||||||
.take_while(|l| !l.starts_with("##"))
|
.take_while(|l| !l.starts_with("##"))
|
||||||
.collect::<Vec<_>>()
|
.collect::<Vec<_>>()
|
||||||
|
@ -74,15 +153,13 @@ fn parse_help(section: &str) -> String {
|
||||||
.to_string()
|
.to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the usage from the "Usage" section in the help file.
|
/// Parses a markdown code block into a usage string
|
||||||
///
|
///
|
||||||
/// The usage is assumed to be surrounded by markdown code fences. It may span
|
/// The code fences are removed and the name of the util is replaced
|
||||||
/// multiple lines. The first word of each line is assumed to be the name of
|
/// with `{}` so that it can be replaced with the appropriate name
|
||||||
/// the util and is replaced by "{}" so that the output of this function can be
|
/// at runtime.
|
||||||
/// used with `uucore::format_usage`.
|
fn parse_usage(content: &str) -> String {
|
||||||
#[proc_macro]
|
content
|
||||||
pub fn help_usage(_input: TokenStream) -> TokenStream {
|
|
||||||
let text: String = parse_help("usage")
|
|
||||||
.strip_suffix("```")
|
.strip_suffix("```")
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.lines()
|
.lines()
|
||||||
|
@ -96,34 +173,74 @@ pub fn help_usage(_input: TokenStream) -> TokenStream {
|
||||||
"{}".to_string()
|
"{}".to_string()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.collect();
|
.collect()
|
||||||
TokenTree::Literal(Literal::string(&text)).into()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Reads a section from the help file of the util as a `str` literal.
|
#[cfg(test)]
|
||||||
///
|
mod tests {
|
||||||
/// It is read verbatim, without parsing or escaping. The name of the help file
|
use super::{parse_help_section, parse_usage};
|
||||||
/// should match the name of the util. I.e. numfmt should have a file called
|
|
||||||
/// `numfmt.md`. By convention, the file should start with a top-level section
|
#[test]
|
||||||
/// with the name of the util. The other sections must start with 2 `#`
|
fn section_parsing() {
|
||||||
/// characters. Capitalization of the sections does not matter. Leading and
|
let input = "\
|
||||||
/// trailing whitespace will be removed. Example:
|
# ls\n\
|
||||||
/// ```md
|
## some section\n\
|
||||||
/// # numfmt
|
This is some section\n\
|
||||||
/// ## About
|
\n\
|
||||||
/// Convert numbers from/to human-readable strings
|
## ANOTHER SECTION
|
||||||
///
|
This is the other section\n\
|
||||||
/// ## Long help
|
with multiple lines\n";
|
||||||
/// This text will be the long help
|
|
||||||
/// ```
|
assert_eq!(
|
||||||
#[proc_macro]
|
parse_help_section("some section", input),
|
||||||
pub fn help_section(input: TokenStream) -> TokenStream {
|
"This is some section"
|
||||||
let input: Vec<TokenTree> = input.into_iter().collect();
|
);
|
||||||
let value = match &input.get(0) {
|
assert_eq!(
|
||||||
Some(TokenTree::Literal(literal)) => literal.to_string(),
|
parse_help_section("another section", input),
|
||||||
_ => panic!("Input to help_section should be a string literal!"),
|
"This is the other section\nwith multiple lines"
|
||||||
};
|
);
|
||||||
let input_str: String = value.parse().unwrap();
|
}
|
||||||
let text = parse_help(&input_str);
|
|
||||||
TokenTree::Literal(Literal::string(&text)).into()
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn section_parsing_panic() {
|
||||||
|
let input = "\
|
||||||
|
# ls\n\
|
||||||
|
## some section\n\
|
||||||
|
This is some section\n\
|
||||||
|
\n\
|
||||||
|
## ANOTHER SECTION
|
||||||
|
This is the other section\n\
|
||||||
|
with multiple lines\n";
|
||||||
|
parse_help_section("non-existent section", input);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn usage_parsing() {
|
||||||
|
let input = "\
|
||||||
|
# ls\n\
|
||||||
|
## Usage\n\
|
||||||
|
```\n\
|
||||||
|
ls -l\n\
|
||||||
|
```\n\
|
||||||
|
## some section\n\
|
||||||
|
This is some section\n\
|
||||||
|
\n\
|
||||||
|
## ANOTHER SECTION
|
||||||
|
This is the other section\n\
|
||||||
|
with multiple lines\n";
|
||||||
|
|
||||||
|
assert_eq!(parse_usage(&parse_help_section("usage", input)), "{} -l",);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
parse_usage(
|
||||||
|
"\
|
||||||
|
```\n\
|
||||||
|
util [some] [options]\n\
|
||||||
|
```\
|
||||||
|
"
|
||||||
|
),
|
||||||
|
"{} [some] [options]"
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue