Browse Source

修改命令解析算法 (#24)

MemoryShore 1 year ago
parent
commit
139fb8f1f5
1 changed files with 58 additions and 13 deletions
  1. 58 13
      src/shell/command/mod.rs

+ 58 - 13
src/shell/command/mod.rs

@@ -42,6 +42,7 @@ pub enum CommandError {
     DirectoryNotFound(String),
     NotDirectory(String),
     NotFile(String),
+    UnclosedQuotation(usize),
 }
 
 impl CommandError {
@@ -74,6 +75,9 @@ impl CommandError {
             CommandError::NotFile(path) => {
                 println!("path is not a file: {}", path)
             }
+            CommandError::UnclosedQuotation(index) => {
+                println!("command exists unclosed quotation at index: {}", index)
+            }
         };
     }
 }
@@ -95,18 +99,59 @@ impl Command {
         });
     }
 
+    fn parse_command_into_fragments(str: String) -> Result<Vec<String>, usize> {
+        let iter = str.chars();
+        let mut fragments: Vec<String> = Vec::new();
+        let mut stack: String = String::with_capacity(str.len());
+        let mut left_quote: char = ' ';
+        let mut left_quote_index: usize = 0;
+        for (index, ch) in iter.enumerate() {
+            //存在未闭合的左引号,此时除能够配对的引号外,任何字符都加入栈中
+            if left_quote != ' ' {
+                if ch == left_quote {
+                    left_quote = ' ';
+                } else {
+                    stack.push(ch);
+                }
+            } else {
+                //不存在未闭合的左引号
+                if ch == '\'' || ch == '\"' {
+                    //字符为引号,记录下来
+                    left_quote = ch;
+                    left_quote_index = index;
+                } else if ch == ' ' {
+                    if !stack.is_empty() {
+                        //字符为空格且栈中不为空,该空格视作命令段之间的分割线
+                        //将栈中字符作为一个命令段加入集合,之后重置栈
+                        fragments.push(stack.to_string());
+                        stack.clear();
+                    }
+                } else {
+                    //其他字符都作为普通字符加入栈中
+                    stack.push(ch);
+                }
+            }
+        }
+        //结束时如果栈不为空
+        if !stack.is_empty() {
+            if left_quote == ' ' {
+                //不存在未闭合的引号,将栈中剩余内容作为命令段加入集合
+                fragments.push(stack.to_string());
+            } else {
+                //存在未闭合的引号,返回此引号的下标
+                return Err(left_quote_index);
+            }
+        }
+        Ok(fragments)
+    }
+
     fn from_string(str: String) -> Result<Command, CommandError> {
-        let regex: Regex = Regex::new(r#"'.*'|".*"|[^\s]+"#).unwrap();
-        let hay = str.clone();
-        let mut iter = regex.captures_iter(hay.as_str()).map(|c| {
-            let str = c.get(0).unwrap().as_str();
-            if str.starts_with(|char| char == '\'' || char == '\"')
-                && str.ends_with(|char| char == '\'' || char == '\"') && str.len() != 1
-            {
-                return str[1..str.len() - 1].to_string();
-            } 
-            return str.to_string();
-        });
+        let iter = Self::parse_command_into_fragments(str);
+        if let Err(index) = iter {
+            return Err(CommandError::UnclosedQuotation(index));
+        }
+        let mut iter = iter.unwrap().into_iter();
+
         let name = iter.next().unwrap();
         let re: Regex = Regex::new(r"\$[\w_]+").unwrap();
         let replacement = |caps: &Captures| -> String {
@@ -138,7 +183,7 @@ impl Command {
             if segment.trim().is_empty() {
                 continue;
             } else {
-                match  Command::from_string(String::from(segment)) {
+                match Command::from_string(String::from(segment)) {
                     Ok(s) => commands.push(s),
                     Err(e) => {
                         CommandError::handle(e);
@@ -147,7 +192,7 @@ impl Command {
             }
         }
 
-        commands 
+        commands
     }
 }