瀏覽代碼

修复后台命令运行时工作路径不正确的问题 (#50)

MemoryShore 5 月之前
父節點
當前提交
0b5cf83b39
共有 3 個文件被更改,包括 24 次插入26 次删除
  1. 3 1
      src/parser.rs
  2. 17 15
      src/shell/mod.rs
  3. 4 10
      src/shell/printer.rs

+ 3 - 1
src/parser.rs

@@ -8,6 +8,8 @@ use std::{
 
 use regex::Regex;
 
+use crate::env::EnvManager;
+
 #[derive(Debug)]
 pub enum Token {
     Word(String),   // 普通的命令或选项
@@ -533,7 +535,7 @@ impl Pipeline {
                     Ok(real_path) => {
                         let mut child_command = std::process::Command::new(real_path);
                         child_command.args(cmd.args.clone());
-                        child_command.current_dir(std::env::current_dir().unwrap());
+                        child_command.current_dir(EnvManager::current_dir());
                         if stdout.is_some() {
                             child_command.stdin(stdout.take().unwrap().as_std());
                         }

+ 17 - 15
src/shell/mod.rs

@@ -9,6 +9,7 @@ use std::{
 };
 
 use crate::{
+    env::EnvManager,
     keycode::{FunctionKeySuffix, SpecialKeycode},
     parser::{Parser, Pipeline},
 };
@@ -31,7 +32,7 @@ pub struct Shell {
     history_commands: Vec<Rc<RefCell<Vec<u8>>>>,
     history_path: String,
     printer: Printer,
-    backend_thread: ThreadManager<Pipeline, Child>,
+    backend_thread: ThreadManager<(String, Vec<Pipeline>), Child>,
 }
 
 impl Shell {
@@ -50,15 +51,18 @@ impl Shell {
         shell
     }
 
-    fn create_backend_thread() -> ThreadManager<Pipeline, Child> {
+    fn create_backend_thread() -> ThreadManager<(String, Vec<Pipeline>), Child> {
         ThreadManager::new(|| {
-            let (p_s, c_r) = std::sync::mpsc::channel::<Pipeline>();
+            let (p_s, c_r) = std::sync::mpsc::channel::<(String, Vec<Pipeline>)>();
             let (c_s, p_r) = std::sync::mpsc::channel::<Child>();
             let map = BuildInCmd::map();
             let func = move || loop {
-                if let Ok(pipeline) = c_r.recv() {
-                    for child in pipeline.execute(map.clone()) {
-                        let _ = c_s.send(child);
+                if let Ok((dir, pipelines)) = c_r.recv() {
+                    std::env::set_current_dir(dir).expect("set current dir failed");
+                    for pipeline in pipelines {
+                        for child in pipeline.execute(map.clone()) {
+                            let _ = c_s.send(child);
+                        }
                     }
                 };
             };
@@ -108,28 +112,26 @@ impl Shell {
     }
 
     fn exec_commands_in_line(&mut self, command_bytes: &Vec<u8>) {
-        // match Command::parse(String::from_utf8(command_bytes.clone()).unwrap()) {
-        //     Ok(commands) => commands
-        //         .into_iter()
-        //         .for_each(|command| self.exec_command(command)),
-        //     Err(e) => CommandError::handle(e),
-        // }
-
         // 解析命令
         let input_command = String::from_utf8(command_bytes.clone()).unwrap();
         let pipelines = Parser::parse(&input_command).unwrap();
 
         let mut foreground_pipelines = Vec::new();
+        let mut backend_pipelines = Vec::new();
 
-        // 后台pipeline发送给子线程执行
         for pipeline in pipelines {
             if pipeline.backend() {
-                let _ = self.backend_thread.send(pipeline);
+                backend_pipelines.push(pipeline);
             } else {
                 foreground_pipelines.push(pipeline);
             }
         }
 
+        // 后台pipeline发送给子线程执行
+        let _ = self
+            .backend_thread
+            .send((EnvManager::current_dir(), backend_pipelines));
+
         crossterm::terminal::disable_raw_mode().expect("failed to disable raw mode");
 
         // 顺序执行所有前台pipeline

+ 4 - 10
src/shell/printer.rs

@@ -9,6 +9,8 @@ use std::{
 
 use colored::Colorize;
 
+use crate::env::EnvManager;
+
 pub struct Printer {
     /// 提示语
     pub prompt: Prompt,
@@ -24,11 +26,7 @@ impl Printer {
             prompt: Prompt {
                 computer_name: "DragonOS".to_string(),
                 user_name: "root".to_string(),
-                path: std::env::current_dir()
-                    .expect("Error getting current directory")
-                    .to_str()
-                    .unwrap()
-                    .to_string(),
+                path: EnvManager::current_dir(),
             },
             buf: Rc::clone(bytes),
             cursor: 0,
@@ -186,11 +184,7 @@ impl Prompt {
     }
 
     pub fn update_path(&mut self) {
-        self.path = std::env::current_dir()
-            .unwrap()
-            .to_str()
-            .unwrap()
-            .to_string();
+        self.path = EnvManager::current_dir()
     }
 }