add command_type to help (#14165)

# Description

This PR adds an indicator when listing subcommands. That indicator tells
whether the command is a plugin, alias, or custom_command.

![image](https://github.com/user-attachments/assets/02889f8a-17b4-4678-bb44-3a487b3d1066)

I changed some of the API to make this work a little easier, namely
`get_signatures()` is now `get_signatures_and_declids()`. It was used in
only one other place (run-external), so I thought it was fine to change
it.

There is a long-standing issue with aliases where they reference the
command name instead of the alias name. This PR doesn't fix that bug.
Example.
```nushell
❯ alias "str fill" = str wrap
```
```nushell
❯ str
... other stuff
Subcommands:

  str wrap (alias) - Alias for `str wrap`
  str wrap (plugin) - Wrap text passed into pipeline.

```


# User-Facing Changes
Slightly different output of subcommands.
This commit is contained in:
Darren Schroeder 2024-10-24 12:06:49 -05:00 committed by GitHub
parent af9c31152a
commit 69d81cc065
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 26 additions and 12 deletions

View file

@ -449,8 +449,8 @@ pub fn command_not_found(
} }
// Try to match the name with the search terms of existing commands. // Try to match the name with the search terms of existing commands.
let signatures = engine_state.get_signatures(false); let signatures = engine_state.get_signatures_and_declids(false);
if let Some(sig) = signatures.iter().find(|sig| { if let Some((sig, _)) = signatures.iter().find(|(sig, _)| {
sig.search_terms sig.search_terms
.iter() .iter()
.any(|term| term.to_folded_case() == name.to_folded_case()) .any(|term| term.to_folded_case() == name.to_folded_case())
@ -463,7 +463,7 @@ pub fn command_not_found(
} }
// Try a fuzzy search on the names of all existing commands. // Try a fuzzy search on the names of all existing commands.
if let Some(cmd) = did_you_mean(signatures.iter().map(|sig| &sig.name), name) { if let Some(cmd) = did_you_mean(signatures.iter().map(|(sig, _)| &sig.name), name) {
// The user is invoking an external command with the same name as a // The user is invoking an external command with the same name as a
// built-in command. Remind them of this. // built-in command. Remind them of this.
if cmd == name { if cmd == name {

View file

@ -2,6 +2,7 @@ use crate::eval_call;
use nu_protocol::{ use nu_protocol::{
ast::{Argument, Call, Expr, Expression, RecordItem}, ast::{Argument, Call, Expr, Expression, RecordItem},
debugger::WithoutDebug, debugger::WithoutDebug,
engine::CommandType,
engine::{Command, EngineState, Stack, UNKNOWN_SPAN_ID}, engine::{Command, EngineState, Stack, UNKNOWN_SPAN_ID},
record, Category, Config, Example, IntoPipelineData, PipelineData, PositionalArg, Signature, record, Category, Config, Example, IntoPipelineData, PipelineData, PositionalArg, Signature,
Span, SpanId, Spanned, SyntaxShape, Type, Value, Span, SpanId, Spanned, SyntaxShape, Type, Value,
@ -112,18 +113,31 @@ fn get_documentation(
// - https://github.com/nushell/nushell/issues/11447 // - https://github.com/nushell/nushell/issues/11447
// - https://github.com/nushell/nushell/issues/11625 // - https://github.com/nushell/nushell/issues/11625
let mut subcommands = vec![]; let mut subcommands = vec![];
let signatures = engine_state.get_signatures(true); let signatures = engine_state.get_signatures_and_declids(true);
for sig in signatures { for (sig, decl_id) in signatures {
let command_type = engine_state.get_decl(decl_id).command_type();
// Don't display removed/deprecated commands in the Subcommands list // Don't display removed/deprecated commands in the Subcommands list
if sig.name.starts_with(&format!("{cmd_name} ")) if sig.name.starts_with(&format!("{cmd_name} "))
&& !matches!(sig.category, Category::Removed) && !matches!(sig.category, Category::Removed)
{ {
// If it's a plugin, alias, or custom command, display that information in the help
if command_type == CommandType::Plugin
|| command_type == CommandType::Alias
|| command_type == CommandType::Custom
{
subcommands.push(format!(
" {help_subcolor_one}{} {help_section_name}({}){RESET} - {}",
sig.name, command_type, sig.description
));
} else {
subcommands.push(format!( subcommands.push(format!(
" {help_subcolor_one}{}{RESET} - {}", " {help_subcolor_one}{}{RESET} - {}",
sig.name, sig.description sig.name, sig.description
)); ));
} }
} }
}
if !subcommands.is_empty() { if !subcommands.is_empty() {
let _ = write!(long_desc, "\n{help_section_name}Subcommands{RESET}:\n"); let _ = write!(long_desc, "\n{help_section_name}Subcommands{RESET}:\n");

View file

@ -820,14 +820,14 @@ impl EngineState {
} }
} }
/// Get signatures of all commands within scope. /// Get signatures of all commands within scope with their decl ids.
pub fn get_signatures(&self, include_hidden: bool) -> Vec<Signature> { pub fn get_signatures_and_declids(&self, include_hidden: bool) -> Vec<(Signature, DeclId)> {
self.get_decls_sorted(include_hidden) self.get_decls_sorted(include_hidden)
.into_iter() .into_iter()
.map(|(_, id)| { .map(|(_, id)| {
let decl = self.get_decl(id); let decl = self.get_decl(id);
self.get_signature(decl).update_from_command(decl) (self.get_signature(decl).update_from_command(decl), id)
}) })
.collect() .collect()
} }