我已经为我的symfony2项目创建了一个控制台命令,我想从控制器执行它而不阻塞控制器输出(在后台)。
通常执行方式如下:
$application = new Application($kernel);
$application->setAutoExit(true);
// AppBundle/Command/UpdateStockCommand.php
$input = new ArrayInput(array(
'command' => 'update:stock',
));
$output = new NullOutput();
$application->run($input, $output);
但是像这样运行,用户必须等待完成任务,这可能需要几分钟。
解决方案是:
$kernel = $this->get('kernel');
$process = new \Symfony\Component\Process\Process('nohup php '. $kernel->getRootDir() .'/console update:stock --env='. $kernel->getEnvironment() .' > /dev/null 2>&1 &');
//$process->start();
$process->run();
没有给出错误,控制器会渲染输出,但不会执行任务。
另一种解决方案是:
exec('/usr/bin/php '.$this->get('kernel')->getRootDir().'/console update:stock --env=dev > /dev/null 2>&1 &');
在此处找到Symfony2 - process launching a symfony2 command 但不适用于我的示例。
答案 0 :(得分:2)
系统中的所有进程都有自己的层次结构。
例如:我们有一个Process A
,在我们运行Process B
之后。如果你杀了Process A
,那么Process B
被杀,因为Process B
是Process A
的孩子。
每个请求(http)Apache为运行PHP代码创建一个新的子进程并将stdoutput返回给客户端(Nginx + PHPFPM的逻辑 - 相同)。在创建子进程(通过Symfony/Process library
)之后,此进程是apache或fpm进程的子进程。完成请求(返回对apache或nginx的响应)后,服务器终止子进程(执行PHP代码)。
在我的项目中,对于运行后台任务,我使用RabbitMQ。
答案 1 :(得分:1)
您可以将输出设置为新的NullOutput。
$output = new NullOutput();
$application->run($input, $output);
但是最好的方法是使用RabbitMqueue
答案 2 :(得分:0)
让我扩展@CerapuStefan的解决方案
exec('bash -c "exec nohup setsid '.$this->get('kernel')->getRootDir().'/console update:stock --env=dev > /dev/null 2>&1 &"');
nohup
很重要