gpt4 book ai didi

windows - 如何防止 Rust std::process:Command 将 .exe 相对路径插入参数?

转载 作者:行者123 更新时间:2023-11-29 08:29:17 26 4
gpt4 key购买 nike

我正在使用 Rust 的 std:process:Command 在 Windows 中调用 robocopy。不幸的是,似乎在执行 robocopy 的某个地方,.exe 的相对路径被插入到每个目录之前。

此外,像 net use 这样的简单调用使用相同的方法,尽管我只测试了不依赖于任何目录的调用。

编辑:更新了测试文件夹,使其名称中包含空格。

编辑 2:代码已用解决方案更新。解决方案是直接调用 robocopy 并在所有内容中单独调用 .arg()。旧的尝试仍在评论中。

而不是发送

cmd.exe /c robocopy "C:\Test\Test 1" "C:\Test\Test 2" /MIR /copy:DAT /MT:32 /Z /R:2 /W:03 /v /LOG:"C:\Test\Test 3\logfile.log"

运行成功,输出有很多错误:

ERROR 123 (0x0000007B) Opening Log File C:\relative\path\where\exe\is\located\"C:\Test\Test 3\logfile.log"
The filename, directory name, or volume label syntax is incorrect.

-------------------------------------------------------------------------------
ROBOCOPY :: Robust File Copy for Windows
-------------------------------------------------------------------------------

Started : Tuesday, January 8, 2019 7:35:41 AM
Source - C:\relative\path\where\exe\is\located\"C:\Test\Test 1"\
Dest - C:\relative\path\where\exe\is\located\"C:\Test\Test 2"\

Files :
Options : /V /S /E /DCOPY:DA /COPY:DAT /PURGE /MIR /Z /MT:32 /R:2 /W:3

------------------------------------------------------------------------------

ERROR : Invalid Parameter #10 : "/LOG:"C:\Test\Test 3\logfile.log""

此问题发生在使用:

  • Rust 版本 1.3.1

  • Windows 10

导致此问题的代码如下所示。要使用它,您需要在 C:\ 中放置一个 Test 文件夹(或将其更改为任意位置),并在该目录中放置一个 Test 1 文件夹,其中包含一些 Testfile.txt 以帮助显示复制的发生,以及用于日志的 Test 3 文件夹(不确定日志是否可以制作自己的文件夹 - 所以为了安全起见,为它预先制作一个!)。基本上你想要:

C:\Test\Test 1\Testfile.txt (name or type of file doesn't matter)
C:\Test\Test 3\

robocopy 应该运行并创建一个 C:\Test\Test 2 文件夹,其中包含 Testfile.txt 并放置一个 C:\Test\Test 3 中的 logfile.log 详细说明了事务。最后,目录应如下所示:

C:\Test\Test 1\Testfile.txt
C:\Test\Test 2\Testfile.txt
C:\Test\Test 3\logfile.log

代码如下:

//-----Import Built-in Libraries (not called crates)-----
use std::process::Command; //use cmd.exe

fn main()
{
let commandOpt1 = "/MIR"; //mirror directories
let commandOpt2 = "/copy:DAT"; //copy attributes
let commandOpt3 = "/MT:32"; //use 32 I/O threads (low CPU still, but better bandwidth utilization)
let commandOpt4 = "/Z"; //idr
let commandOpt5 = "/R:2"; //Retry twice
let commandOpt6 = "/W:03"; //Wait 3 sec between tries
let commandOpt7 = "/v"; //verbose logging
let commandLogStr = "/LOG:\"C:\\Test\\Test 3\\logfile.log\""; //record where the log file goes
let commandLogNoParenthStr = "/LOG:C:\\Test\\Test 3\\logfile.log"; //record where the log file goes
let command = format!("robocopy \"C:\\Test\\Test 1\" \"C:\\Test\\Test 2\" {} {} {} {} {} {} {} {}",
commandOpt1,commandOpt2,commandOpt3,commandOpt4,commandOpt5,commandOpt6,
commandOpt7,commandLogStr); //build the command
let commandStr: &str = &*command; //these two types of strings are
println!("TEST-Command for robocopy:{}",command);

//let m = Command::new("cmd.exe").arg("/c").arg(commandStr).output().unwrap(); //run cmd.exe net use
/*let m = Command::new("cmd.exe")
.arg("/c")
.arg("robocopy")
.arg("\"C:\\Test\\Test 1\"")
.arg("\"C:\\Test\\Test 2\"")
.arg(commandOpt1)
.arg(commandOpt2)
.arg(commandOpt3)
.arg(commandOpt4)
.arg(commandOpt5)
.arg(commandOpt6)
.arg(commandOpt7)
.arg(commandLogStr)
.output().unwrap(); //run cmd.exe net use */
let m = Command::new("robocopy")
.arg("C:\\Test\\Test 1")
.arg("C:\\Test\\Test 2")
.arg(commandOpt1)
.arg(commandOpt2)
.arg(commandOpt3)
.arg(commandOpt4)
.arg(commandOpt5)
.arg(commandOpt6)
.arg(commandOpt7)
.arg(commandLogNoParenthStr )
.output().unwrap(); //run cmd.exe net use */
let mOutput = String::from_utf8_lossy(&m.stdout); //get the output
println!("TEST-cmd output: {}",mOutput);
println!("TEST-cmd status: {}", m.status.success()); //reports success (true or false) of command
println!("TEST-cmd stderr: {}", String::from_utf8_lossy(&m.stderr)); //reports error words of command
let _ = Command::new("cmd.exe").arg("/c").arg("pause").status(); // wait for user input
}

该代码提供了两种选择,一种是在一个 .arg() 添加中调用 robocopy,另一种是使用 .arg()分开。对我来说,它们给出了相同的结果 - 但选项很好!

此外,请注意,我将所有内容作为 &str 发送到 Command 中,但这似乎不是必需的 - Strings 可以是 .arg()也是。

最佳答案

我附近没有 Windows 机器,但我很确定问题出在引号字符 " 中。

CMD.exe 中的转义规则很奇怪:有时需要引号,有时不需要,有时不能使用它们。

在您的情况下,例如使用以下选项选择日志文件(删除 Rust 转义):/LOG:"C:\Test\Test3\logfile.log"。这是一个以引号开头和结尾的文件名。由于它不是以驱动器号或 \ 开头,操作系统认为:肯定是相对路径,让我们寻找 C:\...\"C:\Test ...”!

解决方案很简单:只需删除所有这些引用即可。

我不明白的是你为什么调用 cmd.exe/c 而不是直接调用 robocopy。命令行中的引号用于引导命令行解析器并正确管理带空格的文件名等。但是,如果您直接执行 robocopyCommand::new(),则不需要引用,因为参数已经单独传递。

关于windows - 如何防止 Rust std::process:Command 将 .exe 相对路径插入参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54155214/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com