如何在不将简单的 cli 运行时更改为 php-fpm 的情况下从 Web 管理我的守护进程?
守护进程在操作系统启动后自动启动,并作为没有 php-fpm 池的 cli-应用程序工作。因此,守护生态系统的基本思想是在没有php-fpm池(CLI-SAPI(的情况下工作。
服务器配置:
- Debian 7
- 阿帕奇 2.2
- PHP5-FPM (v 5.4.35( — mod_fastcgi
- 守护程序.php
- daemon_manager.php — 从命令行启动|停止|重新启动|终止守护程序的管理脚本.php。
- daemon_manager_web.php — 用于从浏览器管理守护程序的管理员脚本。
守护进程.php是一个常规的PHP守护进程,如下所示:
<?php
declare(ticks=1);
ini_set("max_execution_time", "0");
ini_set("max_input_time", "0");
set_time_limit(0);
/* Catching signals */
function sig_handler($signo) {
switch ($signo) {
case SIGQUIT:
case SIGTERM:
// some work
pcntl_wait($status);
break;
//...
}
}
pcntl_signal(SIGTERM, 'signal_handler');
pcntl_signal(SIGQUIT, 'signal_handler');
$newpid = pcntl_fork();
if ($newpid == -1) {
throw new Exception('Cannot fork porcess');
} elseif ($newpid) {
print "Starting daemon under pid=$newpid'n";
// ...
exit;
}
问题。
由于 PCNTL 函数无法从 Web 上获得,因此我通过 exec((、shell_exec(( 等函数来管理守护进程。但是当我从浏览器中使用 daemon_manager_web.php 停止并再次启动守护程序时,它通常会启动但在 php-fpm 池下工作。
重新启动前的进程列表:
$ ps aux | grep php
root 5952 0.0 2.9 69008 14952 pts/0 S 14:24 0:00 php /var/www/daemon.php
$ service php5-fpm status
php5-fpm.service - LSB: starts php5-fpm
Loaded: loaded (/etc/init.d/php5-fpm)
Active: active (running) since Fri, 05 Dec 2014 11:28:25 +0200; 11h ago
Process: 1003 ExecStart=/etc/init.d/php5-fpm start (code=exited, status=0/SUCCESS)
CGroup: name=systemd:/system/php5-fpm.service
├ 1627 php-fpm: master process (/etc/php5/fpm/php-fpm.conf)
├ 9562 php-fpm: pool www
├ 9605 php-fpm: pool www
└ 9633 php-fpm: pool www
从浏览器重新启动后的进程列表:
$ service php5-fpm status
php5-fpm.service - LSB: starts php5-fpm
Loaded: loaded (/etc/init.d/php5-fpm)
Active: active (running) since Fri, 05 Dec 2014 11:28:25 +0200; 11h ago
Process: 1003 ExecStart=/etc/init.d/php5-fpm start (code=exited, status=0/SUCCESS)
CGroup: name=systemd:/system/php5-fpm.service
├ 1627 php-fpm: master process (/etc/php5/fpm/php-fpm.conf)
├ 4987 php-fpm: pool www
├ 5040 php-fpm: pool www
├ 9432 php-fpm: pool www
└ 9492 /usr/bin/php /var/www/daemon.php
你不应该以任何方式通过Apache启动守护进程。正确的方法是启动一个这样的守护进程(由,比如说,受监督管理,我在生产中使用它有很好的记录(并打开一个文件套接字 (AF_UNIX(,您可以在其上执行socket_select()
并空闲等待一些触发处理的输入。这样,"接口"(在 Apache 上(只需连接到套接字并写入它。
关于这个主题,我发现用 PHP 编写守护进程非常乏味,你可能想选择一个库来处理你的细节(不能推荐任何(,甚至是另一个更适合守护进程典型事件循环的工具(node.js?