0

我尝试通过 execve() 运行多个命令(或使用简单的输出重定向)。

当我放这个时(当然,在我将这个字符串传递给函数之前,我分成空格并将每个单独的字符放在 char* [] 中):

"bash -c ' /usr/bin/cat /root/script.sh > /root/script1.sh ' "

到 execve() 函数,我有一个错误:

/usr/bin/cat: -c: line 0: 寻找匹配的'''时出现意外的EOF

/usr/bin/cat: -c: 第 1 行:语法错误:文件意外结束

这是我使用execve () 函数运行多个 linux 命令(位于 PATH 中的应用程序)的建议(出于安全原因)

但是这个解决方案并没有像我预期的那样工作。

有什么想法可以解决我的解决方案吗?也许我可以使用 execve() 否则,但我不知道如何..

编辑:添加简化(对不起,由于公司限制,我无法粘贴原始形式)源代码:

int foo(const char *cmdline)
{
    char d[] = "bash -c ' /usr/bin/cat /root/script.sh > /root/script1.sh ' ";

    args = strtok(d, " ");
    counter = 0;
    while (args != NULL)
    {
        cmdline_args[counter++] = args;
        args = strtok(NULL, " ");
    }
    cmdline_args[counter] = '\0';

    switch (pid = fork()) {
    case -1:
        ret = -1;
    case 0: // for execve
        status = execve(cmdline_args[0], cmdline_args, env);
        exit(status);
    default: // for parent pid
        if (waitpid(pid, &status, 0) < 0) {
            // in case when waitpid failed
        }
    }
    return ret;
}
4

2 回答 2

1

由于您的代码是刚刚编写的,我认为它会bash使用参数执行可执行文件:

[ "bash", "-c", "'", "/usr/bin/cat", "/root/script.sh", ">", "/root/script1.sh", "'", 0]

我猜你想要的目标是:

[ "/bin/bash", "-c", "/usr/bin/cat /root/script.sh >/root/script1.sh", NULL]

bash二进制文件不太可能对争论做出很好的反应'。当bash处理您键入的命令时,它会做很多复杂的工作来处理引用字符串的内容,并从中提取实际预期的参数。看起来你可能不得不重复其中的一些工作,如果你真的必须处理近乎任意的命令cmdline(在这种情况下,我会退后一步想'这真的是做 X 的正确方法吗?')。

此外,execve需要二进制文件的完整路径作为其第一个参数;它不搜索PATH.

另外^2:您的标题提到了反引号`,但您的示例代码提到了单个右引号'——您知道它们非常不同,是吗?

于 2015-11-23T11:38:48.670 回答
0

如果这样做对我有用:

cmdline_args[0] = "bash";
cmdline_args[1] = "-c";
cmdline_args[2] = "/usr/bin/cat /root/script.sh > /root/script1.sh";
cmdline_args[3] = NULL;

所以问题是你正在处理整个命令行字符串strtok。因为它bash得到了多个参数(不仅仅是一个应该的 - 整个命令字符串)。bash可能只是解释第一个命令参数,所以你最终会执行'命令......

不过,可能有更好的方法来做到这一点......

于 2015-11-23T11:53:06.260 回答