mirror of
https://github.com/nushell/nushell.git
synced 2025-01-11 08:48:23 +01:00
fix arg parse (#5754)
* fix arg parse * add ut, fix clippy * simplify code * fmt code
This commit is contained in:
parent
c5cb369d8d
commit
2dea9e6f1f
@ -1,6 +1,3 @@
|
|||||||
use std::fs::File;
|
|
||||||
use std::io::{BufRead, BufReader};
|
|
||||||
|
|
||||||
pub fn escape_quote_string(input: &str) -> String {
|
pub fn escape_quote_string(input: &str) -> String {
|
||||||
let mut output = String::with_capacity(input.len() + 2);
|
let mut output = String::with_capacity(input.len() + 2);
|
||||||
output.push('"');
|
output.push('"');
|
||||||
@ -16,97 +13,80 @@ pub fn escape_quote_string(input: &str) -> String {
|
|||||||
output
|
output
|
||||||
}
|
}
|
||||||
|
|
||||||
fn looks_like_flag(input: &str) -> bool {
|
// Escape rules:
|
||||||
if !input.starts_with('-') {
|
// input argument contains ' '(like abc def), we will convert it to `abc def`.
|
||||||
false
|
// input argument contains --version='xx yy', we will convert it to --version=`'xx yy'`
|
||||||
// it does not start with '-'
|
// input argument contains " or \, we will try to escape input.
|
||||||
} else if !input.starts_with("--") {
|
pub fn escape_for_script_arg(input: &str) -> String {
|
||||||
if input.len() > 2
|
// handle for flag, maybe we need to escape the value.
|
||||||
&& input.chars().nth(2).expect("this should never trigger") != '='
|
if input.starts_with("--") {
|
||||||
&& input.chars().nth(2).expect("this should never trigger") != ' '
|
if let Some((arg_name, arg_val)) = input.split_once('=') {
|
||||||
{
|
// only want to escape arg_val.
|
||||||
false
|
let arg_val = if arg_val.contains(' ') {
|
||||||
// while it start with '-', it is not of the form '-x=y' or '-x y'
|
format!("`{}`", arg_val)
|
||||||
|
} else if arg_val.contains('"') || arg_val.contains('\\') {
|
||||||
|
escape_quote_string(arg_val)
|
||||||
} else {
|
} else {
|
||||||
input.len() >= 2
|
arg_val.into()
|
||||||
}
|
};
|
||||||
} else {
|
return format!("{}={}", arg_name, arg_val);
|
||||||
input.len() > 2
|
|
||||||
// it is either a flag --x or a '--'
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn escape_quote_string_when_flags_are_unclear(input: &str) -> String {
|
if input.contains(' ') {
|
||||||
// internal use only. When reading the file for flags goes wrong, revert back to a manual check
|
format!("`{}`", input)
|
||||||
// for flags.
|
} else if input.contains('"') || input.contains('\\') {
|
||||||
let mut output = String::new();
|
escape_quote_string(input)
|
||||||
if !looks_like_flag(input) {
|
|
||||||
output.push('"');
|
|
||||||
for c in input.chars() {
|
|
||||||
if c == '"' || c == '\\' {
|
|
||||||
output.push('\\');
|
|
||||||
}
|
|
||||||
output.push(c);
|
|
||||||
}
|
|
||||||
output.push('"');
|
|
||||||
output
|
|
||||||
} else if input.contains(' ') || input.contains('=') {
|
|
||||||
// this is a flag that requires delicate handling
|
|
||||||
let mut flag_tripped = false;
|
|
||||||
for c in input.chars() {
|
|
||||||
if c == '"' || c == '\\' {
|
|
||||||
output.push('\\');
|
|
||||||
}
|
|
||||||
output.push(c);
|
|
||||||
if (c == ' ' || c == '=') && !flag_tripped {
|
|
||||||
flag_tripped = true;
|
|
||||||
output.push('"');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
output.push('"');
|
|
||||||
output
|
|
||||||
} else {
|
} else {
|
||||||
// this is a normal flag, aka "--x"
|
input.to_string()
|
||||||
String::from(input)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn escape_quote_string_with_file(input: &str, file: &str) -> String {
|
#[cfg(test)]
|
||||||
// use when you want to cross-compare to a file to ensure flags are checked properly
|
mod test {
|
||||||
let file = File::open(file);
|
use super::escape_for_script_arg;
|
||||||
match file {
|
|
||||||
Ok(f) => {
|
#[test]
|
||||||
let lines = BufReader::new(f).lines();
|
fn test_not_extra_quote() {
|
||||||
for line in lines {
|
// check for input arg like this:
|
||||||
let mut flag_start = false;
|
// nu b.nu 8
|
||||||
let mut word = String::new();
|
assert_eq!(escape_for_script_arg("8"), "8".to_string());
|
||||||
let line_or = line.unwrap_or_else(|_| String::from(" "));
|
|
||||||
if line_or.contains('-') {
|
|
||||||
for n in line_or.chars() {
|
|
||||||
if n == '-' {
|
|
||||||
flag_start = true;
|
|
||||||
}
|
}
|
||||||
if n == ' ' || n == ':' || n == ')' {
|
|
||||||
flag_start = false;
|
#[test]
|
||||||
|
fn test_arg_with_flag() {
|
||||||
|
// check for input arg like this:
|
||||||
|
// nu b.nu linux --version=v5.2
|
||||||
|
assert_eq!(escape_for_script_arg("linux"), "linux".to_string());
|
||||||
|
assert_eq!(
|
||||||
|
escape_for_script_arg("--version=v5.2"),
|
||||||
|
"--version=v5.2".to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
// check for input arg like this:
|
||||||
|
// nu b.nu linux --version v5.2
|
||||||
|
assert_eq!(escape_for_script_arg("--version"), "--version".to_string());
|
||||||
|
assert_eq!(escape_for_script_arg("v5.2"), "v5.2".to_string());
|
||||||
}
|
}
|
||||||
if flag_start {
|
|
||||||
word.push(n);
|
#[test]
|
||||||
}
|
fn test_flag_arg_with_values_contains_space() {
|
||||||
}
|
// check for input arg like this:
|
||||||
}
|
// nu b.nu test_ver --version='xx yy' --arch=ghi
|
||||||
if word.contains(input) || {
|
assert_eq!(
|
||||||
let s: Vec<&str> = input.split('=').collect();
|
escape_for_script_arg("--version='xx yy'"),
|
||||||
word.contains(s[0])
|
"--version=`'xx yy'`".to_string()
|
||||||
} {
|
);
|
||||||
return input.to_string();
|
assert_eq!(
|
||||||
}
|
escape_for_script_arg("--arch=ghi"),
|
||||||
}
|
"--arch=ghi".to_string()
|
||||||
let mut final_word = String::new();
|
);
|
||||||
final_word.push('"');
|
}
|
||||||
final_word.push_str(input);
|
|
||||||
final_word.push('"');
|
#[test]
|
||||||
final_word
|
fn test_escape() {
|
||||||
}
|
// check for input arg like this:
|
||||||
_ => escape_quote_string_when_flags_are_unclear(input),
|
// nu b.nu '"'
|
||||||
|
assert_eq!(escape_for_script_arg(r#"""#), r#""\"""#.to_string());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -8,7 +8,7 @@ mod parse_keywords;
|
|||||||
mod parser;
|
mod parser;
|
||||||
mod type_check;
|
mod type_check;
|
||||||
|
|
||||||
pub use deparse::{escape_quote_string, escape_quote_string_with_file};
|
pub use deparse::{escape_for_script_arg, escape_quote_string};
|
||||||
pub use errors::ParseError;
|
pub use errors::ParseError;
|
||||||
pub use flatten::{flatten_block, flatten_expression, flatten_pipeline, FlatShape};
|
pub use flatten::{flatten_block, flatten_expression, flatten_pipeline, FlatShape};
|
||||||
pub use known_external::KnownExternal;
|
pub use known_external::KnownExternal;
|
||||||
|
@ -17,7 +17,7 @@ use nu_cli::{
|
|||||||
};
|
};
|
||||||
use nu_command::{create_default_context, BufferedReader};
|
use nu_command::{create_default_context, BufferedReader};
|
||||||
use nu_engine::{get_full_help, CallExt};
|
use nu_engine::{get_full_help, CallExt};
|
||||||
use nu_parser::{escape_quote_string, escape_quote_string_with_file, parse};
|
use nu_parser::{escape_for_script_arg, escape_quote_string, parse};
|
||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
ast::{Call, Expr, Expression},
|
ast::{Call, Expr, Expression},
|
||||||
engine::{Command, EngineState, Stack, StateWorkingSet},
|
engine::{Command, EngineState, Stack, StateWorkingSet},
|
||||||
@ -93,7 +93,7 @@ fn main() -> Result<()> {
|
|||||||
let mut args = std::env::args().skip(1);
|
let mut args = std::env::args().skip(1);
|
||||||
while let Some(arg) = args.next() {
|
while let Some(arg) = args.next() {
|
||||||
if !script_name.is_empty() {
|
if !script_name.is_empty() {
|
||||||
args_to_script.push(escape_quote_string_with_file(&arg, &script_name));
|
args_to_script.push(escape_for_script_arg(&arg));
|
||||||
} else if arg.starts_with('-') {
|
} else if arg.starts_with('-') {
|
||||||
// Cool, it's a flag
|
// Cool, it's a flag
|
||||||
let flag_value = match arg.as_ref() {
|
let flag_value = match arg.as_ref() {
|
||||||
|
Loading…
Reference in New Issue
Block a user