为什么 exec() 不包含输出?



当我在 shell 上运行命令echo /tmp/stderr{,.pub} | xargs -n 1 ln -sf /dev/stderr && yes | ssh-keygen -t ed25519 -C "" -N "" -f /tmp/stderr > /dev/null; rm /tmp/stderr{,.pub}时,它返回了私钥/公钥对,如下所示(别担心,不使用密钥对(:

blackbox /home/clock # echo /tmp/stderr{,.pub} | xargs -n 1 ln -sf /dev/stderr && yes | ssh-keygen -q -t ed25519 -C "" -N "" -f /tmp/stderr >/dev/null; rm /tmp/stderr{,.pub}
-----BEGIN OPENSSH PRIVATE KEY-----
b3BlbnNzaC1rZXktdjEAAAAABG5vbmUAAAAEbm9uZQAAAAAAAAABAAAAMwAAAAtzc2gtZW
QyNTUxOQAAACA1vZhl2jqtzhEqaqbKwYjLB1OIH8hMPtWB/PWhqeI/QQAAAIj4n5if+J+Y
nwAAAAtzc2gtZWQyNTUxOQAAACA1vZhl2jqtzhEqaqbKwYjLB1OIH8hMPtWB/PWhqeI/QQ
AAAEAtcSI3RLsOo0CXnat4Gs4JENGyDPbGojIT8GU0E+3vUDW9mGXaOq3OESpqpsrBiMsH
U4gfyEw+1YH89aGp4j9BAAAAAAECAwQF
-----END OPENSSH PRIVATE KEY-----
ssh-ed25519 AAAAC3NzaC1lZDI1NTE5AAAAIDW9mGXaOq3OESpqpsrBiMsHU4gfyEw+1YH89aGp4j9B

但是当我在 PHP 中通过exec()运行命令时,它返回了命令本身:

blackbox /home/clock # php -f key.php
Array
(
[0] => /tmp/stderr{,.pub} | xargs -n 1 ln -sf /dev/stderr && yes | ssh-keygen -t ed25519 -C "" -N "" -f /tmp/stderr > /dev/null; rm /tmp/stderr{,.pub}
)

我使用的代码:

<?php
$exec_output = '';
$exec_return = '';
// echo /tmp/stderr{,.pub} | xargs -n 1 ln -sf /dev/stderr && yes | ssh-keygen -t ed25519 -C "" -N "" -f /tmp/stderr > /dev/null; rm /tmp/stderr{,.pub}
$cmd = array('echo', escapeshellarg('/tmp/stderr{,.pub} |'), 'xargs -n 1 ln -sf /dev/stderr', escapeshellarg('&& yes |'), 'ssh-keygen -t ed25519', escapeshellarg('-C "" -N "" -f /tmp/stderr > /dev/null; rm /tmp/stderr{,.pub}'));
exec(implode(' ', $cmd), $exec_output, $exec_return);
print_r($exec_output);
?>

为什么?

PS:我不想将生成的密钥对填充到文件中,即使不在/tmp 中也是如此。

  • PHP: 7.3.13
  • 用户:根
  1. 外壳路径扩展在作为字符串传递时不起作用:

    echo '/tmp/stderr{,.pub}'

    相反,您需要使用不带引号的

    echo /tmp/stderr{,.pub}

  2. 您不能用程序选项本身引用参数,它将被视为一个参数。

  3. 这里没有用户输入,因此无需转义任何内容。
  4. exec只捕获标准输出,而你正在将你的重定向到/dev/null。

更改后的$cmd数组如下所示:

$cmd = array(
'echo',
'/tmp/stderr{,.pub}',
'|',
'xargs -n 1',
'ln -sf',
'/dev/stderr',
'&&',
'yes 2>/dev/null',
'|',
'ssh-keygen',
'-t',
'ed25519',
'-C',
'""',
'-N',
'""',
'-f',
'/tmp/stderr',
'2>&1',
'>/dev/null',
';',
'rm',
'/tmp/stderr{,.pub}'
);

最新更新