Add examples to commands (#1752)

* Pass &dyn WholeStreamCommand to get_help

* Add an optional example to the WholeStreamCommand trait

* Add an example to the alias command
This commit is contained in:
Elichai Turkel 2020-05-11 23:05:44 +03:00 committed by GitHub
parent 42eb658c37
commit c3a066eeb4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 44 additions and 31 deletions

View File

@ -125,7 +125,9 @@ pub(crate) mod wrap;
pub(crate) use autoview::Autoview; pub(crate) use autoview::Autoview;
pub(crate) use cd::Cd; pub(crate) use cd::Cd;
pub(crate) use command::{whole_stream_command, Command, UnevaluatedCallInfo, WholeStreamCommand}; pub(crate) use command::{
whole_stream_command, Command, Example, UnevaluatedCallInfo, WholeStreamCommand,
};
pub(crate) use alias::Alias; pub(crate) use alias::Alias;
pub(crate) use append::Append; pub(crate) use append::Append;

View File

@ -1,4 +1,4 @@
use crate::commands::WholeStreamCommand; use crate::commands::{Example, WholeStreamCommand};
use crate::context::CommandRegistry; use crate::context::CommandRegistry;
use crate::prelude::*; use crate::prelude::*;
use nu_errors::ShellError; use nu_errors::ShellError;
@ -41,6 +41,13 @@ impl WholeStreamCommand for Alias {
) -> Result<OutputStream, ShellError> { ) -> Result<OutputStream, ShellError> {
args.process(registry, alias)?.run() args.process(registry, alias)?.run()
} }
fn examples(&self) -> &[Example] {
&[Example {
description: "Some people prefer to write one letter instead of two",
example: "alias l [x] { ls $x }",
}]
}
} }
pub fn alias( pub fn alias(

View File

@ -350,6 +350,11 @@ impl EvaluatedCommandArgs {
} }
} }
pub struct Example {
pub example: &'static str,
pub description: &'static str,
}
pub trait WholeStreamCommand: Send + Sync { pub trait WholeStreamCommand: Send + Sync {
fn name(&self) -> &str; fn name(&self) -> &str;
@ -368,6 +373,10 @@ pub trait WholeStreamCommand: Send + Sync {
fn is_binary(&self) -> bool { fn is_binary(&self) -> bool {
false false
} }
fn examples(&self) -> &[Example] {
&[]
}
} }
#[derive(Clone)] #[derive(Clone)]
@ -407,7 +416,7 @@ impl Command {
pub fn run(&self, args: CommandArgs, registry: &CommandRegistry) -> OutputStream { pub fn run(&self, args: CommandArgs, registry: &CommandRegistry) -> OutputStream {
if args.call_info.switch_present("help") { if args.call_info.switch_present("help") {
get_help(self.name(), self.usage(), self.signature(), registry).into() get_help(&*self.0, registry).into()
} else { } else {
match self.0.run(args, registry) { match self.0.run(args, registry) {
Ok(stream) => stream, Ok(stream) => stream,
@ -419,6 +428,10 @@ impl Command {
pub fn is_binary(&self) -> bool { pub fn is_binary(&self) -> bool {
self.0.is_binary() self.0.is_binary()
} }
pub fn stream_command(&self) -> &dyn WholeStreamCommand {
&*self.0
}
} }
pub struct FnFilterCommand { pub struct FnFilterCommand {

View File

@ -23,9 +23,6 @@ impl WholeStreamCommand for From {
_args: CommandArgs, _args: CommandArgs,
registry: &CommandRegistry, registry: &CommandRegistry,
) -> Result<OutputStream, ShellError> { ) -> Result<OutputStream, ShellError> {
Ok( Ok(crate::commands::help::get_help(&*self, registry).into())
crate::commands::help::get_help(self.name(), self.usage(), self.signature(), registry)
.into(),
)
} }
} }

View File

@ -86,22 +86,10 @@ fn help(
// Check for a subcommand // Check for a subcommand
let command_name = format!("{} {}", rest[0].item, rest[1].item); let command_name = format!("{} {}", rest[0].item, rest[1].item);
if let Some(command) = registry.get_command(&command_name) { if let Some(command) = registry.get_command(&command_name) {
return Ok(get_help( return Ok(get_help(command.stream_command(), &registry).into());
&command.name(),
&command.usage(),
command.signature(),
&registry,
)
.into());
} }
} else if let Some(command) = registry.get_command(&document.item) { } else if let Some(command) = registry.get_command(&document.item) {
return Ok(get_help( return Ok(get_help(command.stream_command(), &registry).into());
&command.name(),
&command.usage(),
command.signature(),
&registry,
)
.into());
} else { } else {
return Err(ShellError::labeled_error( return Err(ShellError::labeled_error(
"Can't find command (use 'help commands' for full list)", "Can't find command (use 'help commands' for full list)",
@ -142,20 +130,19 @@ You can also learn more at https://www.nushell.sh/book/"#;
} }
} }
#[allow(clippy::cognitive_complexity)]
pub(crate) fn get_help( pub(crate) fn get_help(
cmd_name: &str, cmd: &dyn WholeStreamCommand,
cmd_usage: &str,
cmd_sig: Signature,
registry: &CommandRegistry, registry: &CommandRegistry,
) -> impl Into<OutputStream> { ) -> impl Into<OutputStream> {
let cmd_name = cmd.name();
let signature = cmd.signature();
let mut help = VecDeque::new(); let mut help = VecDeque::new();
let mut long_desc = String::new(); let mut long_desc = String::new();
long_desc.push_str(&cmd_usage); long_desc.push_str(&cmd.usage());
long_desc.push_str("\n"); long_desc.push_str("\n");
let signature = cmd_sig;
let mut subcommands = String::new(); let mut subcommands = String::new();
for name in registry.names() { for name in registry.names() {
if name.starts_with(&format!("{} ", cmd_name)) { if name.starts_with(&format!("{} ", cmd_name)) {
@ -283,6 +270,16 @@ pub(crate) fn get_help(
} }
} }
let examples = cmd.examples();
if !examples.is_empty() {
long_desc.push_str("\nExamples:\n");
}
for example in examples {
long_desc.push_str(" ");
long_desc.push_str(example.description);
long_desc.push_str(&format!("\n > {}\n", example.example));
}
help.push_back(ReturnSuccess::value( help.push_back(ReturnSuccess::value(
UntaggedValue::string(long_desc).into_value(Tag::from((0, cmd_name.len(), None))), UntaggedValue::string(long_desc).into_value(Tag::from((0, cmd_name.len(), None))),
)); ));

View File

@ -23,9 +23,6 @@ impl WholeStreamCommand for To {
_args: CommandArgs, _args: CommandArgs,
registry: &CommandRegistry, registry: &CommandRegistry,
) -> Result<OutputStream, ShellError> { ) -> Result<OutputStream, ShellError> {
Ok( Ok(crate::commands::help::get_help(self, registry).into())
crate::commands::help::get_help(self.name(), self.usage(), self.signature(), registry)
.into(),
)
} }
} }