我在 Linux (Ubuntu 14.04) 上运行一个简单的 apache 网络服务器,使用 perl CGI 脚本处理一些请求。该脚本使用 system
函数启动系统命令,但我希望它立即返回,而不管系统调用的结果如何。
我一直在传递给 system
的标量参数末尾添加一个 & 符号(我知道命令注入(inject)攻击的含义),尽管这样做导致系统命令立即返回,脚本仍然不会退出,直到底层命令完成。
如果我使用来自 perl CGI 的 system
调用触发一个 10 秒休眠的虚拟 ruby 脚本,那么我对 Web 服务器的请求仍然会等待 10 秒才能最终得到响应。我在 system
调用之后放置了一条日志语句,它会在发出 Web 请求时立即出现,因此 system
调用肯定会立即返回,但脚本仍在等待结束。
This question是相似的,但是这两种解决方案都不适合我。
下面是一些示例代码:
#!/usr/bin/perl
use strict;
use warnings;
use CGI;
use Log::Log4perl qw(:easy);
Log::Log4perl->easy_init(
{ level => $DEBUG, file => ">>/var/log/script.log" } );
print "Content-type: application/json\n\n";
my $cgi = CGI->new();
INFO("Executing command...");
system('sudo -u on-behalf-of-user /tmp/test.rb one two &');
INFO("Command initiated - will return now...");
print '{"error":false}';
编辑:
命令调用是使用 sudo -u
执行的,因为 apache 用户 www-data
需要代表脚本所有者执行脚本的权限,并且我已经更新了我的 sudoers为此适当归档。这不是我的问题的原因,因为我还尝试将脚本所有权更改为 www-data
并运行 system("/tmp/test.rb one two &")
但结果是一样的。
编辑 2:
我还尝试将 exit 0
添加到脚本的末尾,但这没有任何区别。脚本是否有可能立即退出,但 apache 服务器会保留响应直到 perl CGI 调用的脚本完成?还是可能是操作系统的某些设置或配置导致了问题?
编辑 3:
直接从终端运行 perl CGI 脚本可以正常工作。 perl 脚本立即结束,因此这不是 Perl 的固有问题。这可能只意味着 Apache Web 服务器会保留请求,直到从 system
调用的命令完成。为什么?
最佳答案
Web 服务器创建一个管道来接收响应。它在完成请求之前等待管道到达 EOF。当写入器句柄的所有副本都关闭时,管道到达 EOF。
管道的写入端被设置为 child 的 STDOUT。该文件句柄被复制为 shell 的 STDOUT,并再次复制到 mycmd
的 STDOUT。因此,即使 CGI 脚本和 shell 结束并因此关闭了文件句柄的末端,mycmd
仍然保持句柄打开,因此 Web 服务器仍在等待响应完成。
您只需关闭管道写入端的最后一个句柄即可。或者更准确地说,您可以通过将不同的句柄附加到 mycmd
STDOUT 来避免首先创建它。
mycmd arg1 arg2 </dev/null >/dev/null 2>&1 &
关于linux - 结束 perl 脚本而不等待系统调用返回,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28126351/