fix arg parse (#5754)

* fix arg parse

* add ut, fix clippy

* simplify code

* fmt code
This commit is contained in:
WindSoilder 2022-06-11 16:52:31 +08:00 committed by GitHub
parent c5cb369d8d
commit 2dea9e6f1f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 72 additions and 92 deletions

View File

@ -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());
} }
} }

View File

@ -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;

View File

@ -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() {