nushell/crates/nu-protocol/src/signature.rs

567 lines
15 KiB
Rust
Raw Normal View History

use serde::Deserialize;
use serde::Serialize;
2021-09-02 22:58:15 +00:00
use crate::ast::Call;
2022-03-07 20:08:56 +00:00
use crate::ast::Expression;
2021-09-02 18:21:37 +00:00
use crate::engine::Command;
2021-10-25 06:31:39 +00:00
use crate::engine::EngineState;
use crate::engine::Stack;
2021-09-02 08:25:22 +00:00
use crate::BlockId;
2021-10-25 04:01:02 +00:00
use crate::PipelineData;
use crate::ShellError;
2021-09-02 08:25:22 +00:00
use crate::SyntaxShape;
2021-09-02 01:29:43 +00:00
use crate::VarId;
2021-07-01 22:40:08 +00:00
2022-03-07 20:08:56 +00:00
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
2021-07-01 22:40:08 +00:00
pub struct Flag {
pub long: String,
pub short: Option<char>,
pub arg: Option<SyntaxShape>,
pub required: bool,
pub desc: String,
2022-03-07 20:08:56 +00:00
2021-07-23 21:19:30 +00:00
// For custom commands
pub var_id: Option<VarId>,
2022-03-07 20:08:56 +00:00
pub default_value: Option<Expression>,
2021-07-01 22:40:08 +00:00
}
2022-03-07 20:08:56 +00:00
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
2021-07-01 22:40:08 +00:00
pub struct PositionalArg {
pub name: String,
pub desc: String,
pub shape: SyntaxShape,
2022-03-07 20:08:56 +00:00
2021-07-23 21:19:30 +00:00
// For custom commands
pub var_id: Option<VarId>,
2022-03-07 20:08:56 +00:00
pub default_value: Option<Expression>,
2021-07-01 22:40:08 +00:00
}
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
pub enum Category {
Default,
Conversions,
Core,
Date,
Env,
Experimental,
FileSystem,
Filters,
Formats,
Math,
Network,
Random,
Platform,
2021-11-26 08:00:57 +00:00
Shells,
Strings,
System,
Viewers,
Hash,
Generators,
Chart,
Custom(String),
Deprecated,
}
impl std::fmt::Display for Category {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let msg = match self {
Category::Default => "default",
Category::Conversions => "conversions",
Category::Core => "core",
Category::Date => "date",
Category::Env => "env",
Category::Experimental => "experimental",
Category::FileSystem => "filesystem",
Category::Filters => "filters",
Category::Formats => "formats",
Category::Math => "math",
Category::Network => "network",
Category::Random => "random",
Category::Platform => "platform",
2021-11-26 08:00:57 +00:00
Category::Shells => "shells",
Category::Strings => "strings",
Category::System => "system",
Category::Viewers => "viewers",
Category::Hash => "hash",
Category::Generators => "generators",
Category::Chart => "chart",
Category::Custom(name) => name,
Category::Deprecated => "deprecated",
};
write!(f, "{}", msg)
}
}
#[derive(Clone, Debug, Serialize, Deserialize)]
2021-07-01 22:40:08 +00:00
pub struct Signature {
pub name: String,
pub usage: String,
pub extra_usage: String,
pub search_terms: Vec<String>,
2021-07-01 22:40:08 +00:00
pub required_positional: Vec<PositionalArg>,
pub optional_positional: Vec<PositionalArg>,
pub rest_positional: Option<PositionalArg>,
pub named: Vec<Flag>,
pub is_filter: bool,
2021-10-09 16:10:46 +00:00
pub creates_scope: bool,
// Signature category used to classify commands stored in the list of declarations
pub category: Category,
2019-05-28 06:45:18 +00:00
}
impl PartialEq for Signature {
fn eq(&self, other: &Self) -> bool {
self.name == other.name
&& self.usage == other.usage
2021-07-01 22:40:08 +00:00
&& self.required_positional == other.required_positional
&& self.optional_positional == other.optional_positional
&& self.rest_positional == other.rest_positional
&& self.is_filter == other.is_filter
}
}
impl Eq for Signature {}
Restructure and streamline token expansion (#1123) Restructure and streamline token expansion The purpose of this commit is to streamline the token expansion code, by removing aspects of the code that are no longer relevant, removing pointless duplication, and eliminating the need to pass the same arguments to `expand_syntax`. The first big-picture change in this commit is that instead of a handful of `expand_` functions, which take a TokensIterator and ExpandContext, a smaller number of methods on the `TokensIterator` do the same job. The second big-picture change in this commit is fully eliminating the coloring traits, making coloring a responsibility of the base expansion implementations. This also means that the coloring tracer is merged into the expansion tracer, so you can follow a single expansion and see how the expansion process produced colored tokens. One side effect of this change is that the expander itself is marginally more error-correcting. The error correction works by switching from structured expansion to `BackoffColoringMode` when an unexpected token is found, which guarantees that all spans of the source are colored, but may not be the most optimal error recovery strategy. That said, because `BackoffColoringMode` only extends as far as a closing delimiter (`)`, `]`, `}`) or pipe (`|`), it does result in fairly granular correction strategy. The current code still produces an `Err` (plus a complete list of colored shapes) from the parsing process if any errors are encountered, but this could easily be addressed now that the underlying expansion is error-correcting. This commit also colors any spans that are syntax errors in red, and causes the parser to include some additional information about what tokens were expected at any given point where an error was encountered, so that completions and hinting could be more robust in the future. Co-authored-by: Jonathan Turner <jonathandturner@users.noreply.github.com> Co-authored-by: Andrés N. Robalino <andres@androbtech.com>
2020-01-21 22:45:03 +00:00
impl Signature {
2021-07-01 22:40:08 +00:00
pub fn new(name: impl Into<String>) -> Signature {
2021-10-13 17:53:27 +00:00
// default help flag
let flag = Flag {
long: "help".into(),
short: Some('h'),
arg: None,
desc: "Display this help message".into(),
required: false,
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-10-13 17:53:27 +00:00
};
2019-11-18 03:12:37 +00:00
Signature {
Add Range and start Signature support This commit contains two improvements: - Support for a Range syntax (and a corresponding Range value) - Work towards a signature syntax Implementing the Range syntax resulted in cleaning up how operators in the core syntax works. There are now two kinds of infix operators - tight operators (`.` and `..`) - loose operators Tight operators may not be interspersed (`$it.left..$it.right` is a syntax error). Loose operators require whitespace on both sides of the operator, and can be arbitrarily interspersed. Precedence is left to right in the core syntax. Note that delimited syntax (like `( ... )` or `[ ... ]`) is a single token node in the core syntax. A single token node can be parsed from beginning to end in a context-free manner. The rule for `.` is `<token node>.<member>`. The rule for `..` is `<token node>..<token node>`. Loose operators all have the same syntactic rule: `<token node><space><loose op><space><token node>`. The second aspect of this pull request is the beginning of support for a signature syntax. Before implementing signatures, a necessary prerequisite is for the core syntax to support multi-line programs. That work establishes a few things: - `;` and newlines are handled in the core grammar, and both count as "separators" - line comments begin with `#` and continue until the end of the line In this commit, multi-token productions in the core grammar can use separators interchangably with spaces. However, I think we will ultimately want a different rule preventing separators from occurring before an infix operator, so that the end of a line is always unambiguous. This would avoid gratuitous differences between modules and repl usage. We already effectively have this rule, because otherwise `x<newline> | y` would be a single pipeline, but of course that wouldn't work.
2019-12-04 21:14:52 +00:00
name: name.into(),
2019-11-18 03:12:37 +00:00
usage: String::new(),
extra_usage: String::new(),
search_terms: vec![],
2021-07-01 22:40:08 +00:00
required_positional: vec![],
optional_positional: vec![],
rest_positional: None,
2021-10-13 17:53:27 +00:00
named: vec![flag],
2021-07-01 22:40:08 +00:00
is_filter: false,
2021-10-09 16:10:46 +00:00
creates_scope: false,
category: Category::Default,
2021-07-01 22:40:08 +00:00
}
}
2019-08-02 19:15:07 +00:00
pub fn build(name: impl Into<String>) -> Signature {
Signature::new(name.into())
2019-07-24 04:10:48 +00:00
}
/// Add a description to the signature
pub fn usage(mut self, msg: impl Into<String>) -> Signature {
self.usage = msg.into();
self
}
/// Add an extra description to the signature
pub fn extra_usage(mut self, msg: impl Into<String>) -> Signature {
self.extra_usage = msg.into();
self
}
/// Add search terms to the signature
pub fn search_terms(mut self, terms: Vec<String>) -> Signature {
self.search_terms = terms;
self
}
/// Update signature's fields from a Command trait implementation
pub fn update_from_command(mut self, command: &dyn Command) -> Signature {
self.search_terms = command
.search_terms()
.into_iter()
.map(|term| term.to_string())
.collect();
self.extra_usage = command.extra_usage().to_string();
self.usage = command.usage().to_string();
self
}
/// Add a required positional argument to the signature
2019-10-28 05:15:35 +00:00
pub fn required(
mut self,
name: impl Into<String>,
2021-07-01 22:40:08 +00:00
shape: impl Into<SyntaxShape>,
desc: impl Into<String>,
) -> Signature {
self.required_positional.push(PositionalArg {
name: name.into(),
desc: desc.into(),
shape: shape.into(),
2021-07-23 21:19:30 +00:00
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-01 22:40:08 +00:00
});
2019-07-24 04:10:48 +00:00
self
}
2021-07-01 22:40:08 +00:00
/// Add a required positional argument to the signature
pub fn optional(
mut self,
name: impl Into<String>,
shape: impl Into<SyntaxShape>,
desc: impl Into<String>,
) -> Signature {
self.optional_positional.push(PositionalArg {
name: name.into(),
desc: desc.into(),
shape: shape.into(),
2021-07-23 21:19:30 +00:00
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-01 22:40:08 +00:00
});
self
}
2021-09-07 03:37:02 +00:00
pub fn rest(
mut self,
name: &str,
shape: impl Into<SyntaxShape>,
desc: impl Into<String>,
) -> Signature {
2021-07-30 03:26:06 +00:00
self.rest_positional = Some(PositionalArg {
2021-09-07 03:37:02 +00:00
name: name.into(),
2021-07-30 03:26:06 +00:00
desc: desc.into(),
shape: shape.into(),
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-30 03:26:06 +00:00
});
2019-07-24 04:10:48 +00:00
self
}
/// Add an optional named flag argument to the signature
2019-10-28 05:15:35 +00:00
pub fn named(
mut self,
name: impl Into<String>,
2021-07-01 22:40:08 +00:00
shape: impl Into<SyntaxShape>,
desc: impl Into<String>,
short: Option<char>,
) -> Signature {
2021-09-04 07:45:49 +00:00
let (name, s) = self.check_names(name, short);
2021-07-01 22:40:08 +00:00
self.named.push(Flag {
2021-09-04 08:19:07 +00:00
long: name,
2021-07-01 22:40:08 +00:00
short: s,
arg: Some(shape.into()),
required: false,
desc: desc.into(),
2021-07-23 21:19:30 +00:00
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-01 22:40:08 +00:00
});
2019-08-02 19:15:07 +00:00
self
}
/// Add a required named flag argument to the signature
2019-08-02 19:15:07 +00:00
pub fn required_named(
mut self,
name: impl Into<String>,
2021-07-01 22:40:08 +00:00
shape: impl Into<SyntaxShape>,
desc: impl Into<String>,
short: Option<char>,
) -> Signature {
2021-09-04 07:45:49 +00:00
let (name, s) = self.check_names(name, short);
2021-07-01 22:40:08 +00:00
self.named.push(Flag {
2021-09-04 08:19:07 +00:00
long: name,
2021-07-01 22:40:08 +00:00
short: s,
arg: Some(shape.into()),
required: true,
desc: desc.into(),
2021-07-23 21:19:30 +00:00
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-01 22:40:08 +00:00
});
2019-08-02 19:15:07 +00:00
self
}
/// Add a switch to the signature
pub fn switch(
mut self,
name: impl Into<String>,
desc: impl Into<String>,
short: Option<char>,
) -> Signature {
2021-09-04 07:45:49 +00:00
let (name, s) = self.check_names(name, short);
2021-07-01 22:40:08 +00:00
self.named.push(Flag {
2021-09-04 08:19:07 +00:00
long: name,
2021-07-01 22:40:08 +00:00
short: s,
arg: None,
required: false,
desc: desc.into(),
2021-07-23 21:19:30 +00:00
var_id: None,
2022-03-07 20:08:56 +00:00
default_value: None,
2021-07-01 22:40:08 +00:00
});
2021-09-04 07:45:49 +00:00
2021-07-01 22:40:08 +00:00
self
}
/// Changes the signature category
pub fn category(mut self, category: Category) -> Signature {
self.category = category;
self
}
2021-10-09 16:10:46 +00:00
/// Sets that signature will create a scope as it parses
pub fn creates_scope(mut self) -> Signature {
self.creates_scope = true;
self
}
2022-01-03 23:14:33 +00:00
pub fn call_signature(&self) -> String {
let mut one_liner = String::new();
one_liner.push_str(&self.name);
one_liner.push(' ');
// Note: the call signature needs flags first because on the nu commandline,
// flags will precede the script file name. Flags for internal commands can come
// either before or after (or around) positional parameters, so there isn't a strong
// preference, so we default to the more constrained example.
if self.named.len() > 1 {
one_liner.push_str("{flags} ");
}
2022-01-03 23:14:33 +00:00
for positional in &self.required_positional {
one_liner.push_str(&get_positional_short_name(positional, true));
}
for positional in &self.optional_positional {
one_liner.push_str(&get_positional_short_name(positional, false));
}
if let Some(rest) = &self.rest_positional {
one_liner.push_str(&format!("...{}", get_positional_short_name(rest, false)));
2022-01-03 23:14:33 +00:00
}
// if !self.subcommands.is_empty() {
// one_liner.push_str("<subcommand> ");
// }
one_liner
}
2021-07-01 22:40:08 +00:00
/// Get list of the short-hand flags
pub fn get_shorts(&self) -> Vec<char> {
2021-09-04 07:45:49 +00:00
self.named.iter().filter_map(|f| f.short).collect()
}
/// Get list of the long-hand flags
2021-09-04 08:10:31 +00:00
pub fn get_names(&self) -> Vec<&str> {
self.named.iter().map(|f| f.long.as_str()).collect()
2021-09-04 07:45:49 +00:00
}
/// Checks if short or long are already present
/// Panics if one of them is found
fn check_names(&self, name: impl Into<String>, short: Option<char>) -> (String, Option<char>) {
let s = short.map(|c| {
debug_assert!(
!self.get_shorts().contains(&c),
"There may be duplicate short flags, such as -h"
);
c
});
2021-09-04 07:45:49 +00:00
let name = {
2021-09-04 08:10:31 +00:00
let name: String = name.into();
2021-09-04 07:45:49 +00:00
debug_assert!(
2021-09-04 08:10:31 +00:00
!self.get_names().contains(&name.as_str()),
2021-09-04 07:45:49 +00:00
"There may be duplicate name flags, such as --help"
);
name
};
(name, s)
2021-07-01 22:40:08 +00:00
}
pub fn get_positional(&self, position: usize) -> Option<PositionalArg> {
if position < self.required_positional.len() {
self.required_positional.get(position).cloned()
} else if position < (self.required_positional.len() + self.optional_positional.len()) {
self.optional_positional
.get(position - self.required_positional.len())
.cloned()
} else {
self.rest_positional.clone()
}
}
2021-07-07 22:55:46 +00:00
pub fn num_positionals(&self) -> usize {
2021-07-24 05:57:17 +00:00
let mut total = self.required_positional.len() + self.optional_positional.len();
for positional in &self.required_positional {
2021-07-29 22:56:51 +00:00
if let SyntaxShape::Keyword(..) = positional.shape {
// Keywords have a required argument, so account for that
total += 1;
2021-07-24 05:57:17 +00:00
}
}
for positional in &self.optional_positional {
2021-07-29 22:56:51 +00:00
if let SyntaxShape::Keyword(..) = positional.shape {
// Keywords have a required argument, so account for that
total += 1;
2021-07-24 05:57:17 +00:00
}
}
total
}
pub fn num_positionals_after(&self, idx: usize) -> usize {
let mut total = 0;
2021-09-04 07:59:38 +00:00
for (curr, positional) in self.required_positional.iter().enumerate() {
2021-07-24 05:57:17 +00:00
match positional.shape {
SyntaxShape::Keyword(..) => {
// Keywords have a required argument, so account for that
if curr > idx {
total += 2;
}
}
_ => {
if curr > idx {
total += 1;
}
}
}
}
total
2021-07-07 22:55:46 +00:00
}
2021-07-01 22:40:08 +00:00
/// Find the matching long flag
pub fn get_long_flag(&self, name: &str) -> Option<Flag> {
for flag in &self.named {
if flag.long == name {
return Some(flag.clone());
}
}
None
}
/// Find the matching long flag
pub fn get_short_flag(&self, short: char) -> Option<Flag> {
for flag in &self.named {
if let Some(short_flag) = &flag.short {
if *short_flag == short {
return Some(flag.clone());
}
}
}
None
Add --help for commands (#1226) * WIP --help works for PerItemCommands. * De-linting * Add more comments (#1228) * Add some more docs * More docs * More docs * More docs (#1229) * Add some more docs * More docs * More docs * Add more docs * External commands: wrap values that contain spaces in quotes (#1214) (#1220) * External commands: wrap values that contain spaces in quotes (#1214) * Add fn's argument_contains_whitespace & add_quotes (#1214) * Fix formatting with cargo fmt * Don't wrap argument in quotes when $it is already quoted (#1214) * Implement --help for internal commands * Externals now spawn independently. (#1230) This commit changes the way we shell out externals when using the `"$it"` argument. Also pipes per row to an external's stdin if no `"$it"` argument is present for external commands. Further separation of logic (preparing the external's command arguments, getting the data for piping, emitting values, spawning processes) will give us a better idea for lower level details regarding external commands until we can find the right abstractions for making them more generic and unify within the pipeline calling logic of Nu internal's and external's. * Poll externals quicker. (#1231) * WIP --help works for PerItemCommands. * De-linting * Implement --help for internal commands * Make having --help the default * Update test to include new default switch Co-authored-by: Jonathan Turner <jonathandturner@users.noreply.github.com> Co-authored-by: Koenraad Verheyden <mail@koenraadverheyden.com> Co-authored-by: Andrés N. Robalino <andres@androbtech.com>
2020-01-17 22:46:18 +00:00
}
/// Set the filter flag for the signature
2019-08-02 19:15:07 +00:00
pub fn filter(mut self) -> Signature {
self.is_filter = true;
self
}
2021-09-02 08:25:22 +00:00
/// Create a placeholder implementation of Command as a way to predeclare a definition's
/// signature so other definitions can see it. This placeholder is later replaced with the
/// full definition in a second pass of the parser.
pub fn predeclare(self) -> Box<dyn Command> {
Box::new(Predeclaration { signature: self })
}
/// Combines a signature and a block into a runnable block
pub fn into_block_command(self, block_id: BlockId) -> Box<dyn Command> {
Box::new(BlockCommand {
signature: self,
block_id,
})
}
2021-07-01 22:40:08 +00:00
}
2021-10-25 04:01:02 +00:00
#[derive(Clone)]
2021-09-02 08:25:22 +00:00
struct Predeclaration {
signature: Signature,
}
impl Command for Predeclaration {
fn name(&self) -> &str {
&self.signature.name
}
fn signature(&self) -> Signature {
self.signature.clone()
}
fn usage(&self) -> &str {
&self.signature.usage
2021-07-29 22:56:51 +00:00
}
2021-09-02 18:21:37 +00:00
2021-09-02 22:58:15 +00:00
fn run(
&self,
2021-10-25 06:31:39 +00:00
_engine_state: &EngineState,
_stack: &mut Stack,
2021-09-02 22:58:15 +00:00
_call: &Call,
2021-10-25 04:01:02 +00:00
_input: PipelineData,
) -> Result<PipelineData, crate::ShellError> {
2021-09-02 18:21:37 +00:00
panic!("Internal error: can't run a predeclaration without a body")
2021-07-29 22:56:51 +00:00
}
}
2022-01-03 23:14:33 +00:00
fn get_positional_short_name(arg: &PositionalArg, is_required: bool) -> String {
match &arg.shape {
SyntaxShape::Keyword(name, ..) => {
if is_required {
format!("{} <{}> ", String::from_utf8_lossy(name), arg.name)
} else {
format!("({} <{}>) ", String::from_utf8_lossy(name), arg.name)
}
}
_ => {
if is_required {
format!("<{}> ", arg.name)
} else {
format!("({}) ", arg.name)
}
}
}
}
2021-10-25 04:01:02 +00:00
#[derive(Clone)]
2021-09-02 08:25:22 +00:00
struct BlockCommand {
signature: Signature,
block_id: BlockId,
}
impl Command for BlockCommand {
fn name(&self) -> &str {
&self.signature.name
}
fn signature(&self) -> Signature {
2021-09-06 02:20:02 +00:00
self.signature.clone()
2021-09-02 08:25:22 +00:00
}
fn usage(&self) -> &str {
&self.signature.usage
}
2021-09-02 18:21:37 +00:00
2021-09-02 22:58:15 +00:00
fn run(
&self,
2021-10-25 06:31:39 +00:00
_engine_state: &EngineState,
_stack: &mut Stack,
2021-09-02 22:58:15 +00:00
_call: &Call,
2021-10-25 04:01:02 +00:00
_input: PipelineData,
) -> Result<crate::PipelineData, crate::ShellError> {
Err(ShellError::GenericError(
"Internal error: can't run custom command with 'run', use block_id".to_string(),
"".to_string(),
None,
None,
Vec::new(),
))
2021-09-02 18:21:37 +00:00
}
2021-09-05 23:16:27 +00:00
fn get_block_id(&self) -> Option<BlockId> {
2021-09-02 18:21:37 +00:00
Some(self.block_id)
}
2019-07-24 04:10:48 +00:00
}