协程 shell_exec 如何捕获标准错误流

简介: Swoole提供的System::exec()行为上与PHP的shell_exec是完全一致的,我们写一个shell_exec的同步阻塞版本,执行后发现同样拿不到标准错误流输出的内容,会被直接打印到屏幕。那么如何解决这个问题呢?答案就是使用proc_open+hook实现。

实例代码



Swoole\Runtime::setHookFlags(SWOOLE_HOOK_ALL);
Swoole\Coroutine\run(function () {
    $descriptorspec = array(
        0 => array("pipe", "r"),
        1 => array("pipe", "w"),
        2 => array("pipe", "w"),
    );
    $process = proc_open('unknown', $descriptorspec, $pipes);
    var_dump($pipes);
    var_dump(fread($pipes[2], 8192));
    $return_value = proc_close($process);
    echo "command returned $return_value\n";
});

使用proc_open,传入了3个描述信息:

  • fd 为 0的流是标准输入,可以在主进程内向这个流写入数据,子进程就可以得到数据
  • fd 为 1的流是标准输出,这里可以得到执行命令的输出内容
  • fd 为 2的 pipe stream 就是 stderr ,读取 stderr 就能拿到错误信息输出

使用fread就可以拿到标准错误流输出的内容。


htf@htf-ThinkPad-T470p:~/workspace/swoole/examples/coroutine$ php proc_open.php
array(3) {
  [0]=>
  resource(4) of type (stream)
  [1]=>
  resource(5) of type (stream)
  [2]=>
  resource(6) of type (stream)
}
string(26) "sh: 1: unknown: not found
"
command returned 32512
htf@htf-ThinkPad-T470p:~/workspace/swoole/examples/coroutine$
目录
相关文章
|
4月前
|
Shell Linux C语言
Linux中执行Shell的函数(popen,system,exec)介绍:分享一些常用的执行Shell的函数及其相关编程技巧和经验
Linux中执行Shell的函数(popen,system,exec)介绍:分享一些常用的执行Shell的函数及其相关编程技巧和经验
113 0
|
Shell
Shell while 语法(: 死循环)
Shell while 语法(: 死循环)
265 0
|
Shell 程序员
shell中的信号捕获trap(shell 进阶)
shell中的信号捕获trap(shell 进阶)
131 0
shell中的信号捕获trap(shell 进阶)
|
Shell Linux
Shell-使用&和wait让你的脚本并行执行
Shell-使用&和wait让你的脚本并行执行
325 0
|
存储 Shell 测试技术
shell wait 等待命令
shell wait 等待命令
shell wait 等待命令
|
Shell 测试技术 Python
调用 subprocess 时小心 shell=True
记录这个 lessons 学习,调用 subprocess 时特别是包含复杂命令或参数的情况,最好传入命令列表,而不要使用 shell=True 。这样可以避免很多命令行解析引起的问题。
|
Shell C语言 开发者
Shell脚本信号捕获|学习笔记
快速学习Shell脚本信号捕获
Shell脚本信号捕获|学习笔记
|
Shell Linux Go
在 Go 语言中使用 exec 包执行 Shell 命令(上)
exec 是 os 包中的一个子包,它可用于使用 Go 运行外部命令。Go exec 命令教程展示了如何在 Golang 中执行 shell 命令和程序。
|
Shell Go
在 Go 语言中使用 exec 包执行 Shell 命令(下)
exec 是 os 包中的一个子包,它可用于使用 Go 运行外部命令。Go exec 命令教程展示了如何在 Golang 中执行 shell 命令和程序。