使用 SWOOLE 实现进程的守护(一)

微麦PHP 2019-11-03

一、 进程守护使用场景。

后端经常会有类似这样的场景,某个脚本,需要不断的重复运行,这个时候,最好有一个守护程序,帮助我们不断地自动地拉起这些脚本进程,让它自动地重复运行。

在 Linux/Unix 系统下,supervisor 就是使用 python 开发的一个优秀的进程管理工具,本文尝试使用 php 来实现类似的进程管理工具。

二、swoole 的进程管理模块。

php 的 swoole 扩展有一个进程管理模块,官方文档见:swoole 进程管理模块

参考 supervisor 的实现方式,被守护的进程是作为 supervisor 的子进程来启动的,supervisor 通过监听子进程的信号,可实现对子进程的自动重启等功能。而 swoole 的进程管理模块就提供了进程间通信的功能,可以实现对子进程的自动重启功能。

三、第一个进程守护程序。

要实现对子程序的守护,需要做到两点:

  1. 程序需要监听到子进程的结束信号,以便于重新拉起新的子进程。
  2. 子进程的运行环境需要独立于父进程。

swoole 进程管理模块提供了一个 bool Process->exec(string $execfile, array $args) 方法,让子进程蜕变成另一个系统调用程序,同时还能保证父进程与当前进程仍然是父子进程关系。

再通过 array Process::wait(bool $blocking = true) 方法,来等待子进程的退出信号。

下面是使用 swoole 启动子进程,并回收子进程资源的示例代码:

<?php
use Swoole\Process;

$php = "/usr/bin/env php";
$script = dirname(__DIR__) . "/task.php";
$command = "{$php} {$script}";

$process = new Process(function (Process $worker) use ($command) {
    $worker->exec('/bin/sh', ['-c', $command]);
});
$pid = $process->start();

printf("启动子进程 {$pid}\n");

while ($ret = Process::wait()) {
    $pid = intval($ret["pid"] ?? 0);
    printf("子进程 {$pid} 结束\n");
}

代码解析:
$command 变量表示需要子进程脚本,通过 exec() 方法来启动成一个子进程的方式运行,再通过 Process::wait() 访求来等待 $command 这个子进程脚本结束,并回收进程资源。

那么,只要在收到子进程的结束信号后,再起一个相同的子进程脚本,即可实现对子进程的守护了。于是,第一个守护子进程的程序实现代码:

<?php
use Swoole\Process;

$php = "/usr/bin/env php";
$script = dirname(__DIR__) . "/task.php";
$command = "{$php} {$script}";

do {
    $process = new Process(function (Process $worker) use ($command) {
        $worker->exec('/bin/sh', ['-c', $command]);
    });
    $pid = $process->start();

    printf("启动子进程 {$pid}\n");
} while (Process::wait());

代码解析:
这段代码只将启动子进程的逻辑加到一个死循环中,好让这个子进程脚本能够不断的重启。

四、封装成类

为了方便重用这段代码,可以将这段代码封装成一个简单的类:

<?php
namespace App;

use Swoole\Process;

class Daemon
{
    /** @var string */
    private $command;

    public function __construct(string $command)
    {
        $this->command = $command;
    }

    public function run()
    {
        do {
            $process = new Process(function (Process $worker) {
                $worker->exec('/bin/sh', ['-c', $this->command]);
            });
            $pid = $process->start();
        } while (Process::wait());
    }

}

那么,这个 Daemon 类的使用方式如下:

<?php

use App\Daemon;

$php = "/usr/bin/env php";
$script = dirname(__DIR__) . "/task.php";
$command = "{$php} {$script}";

$daemon = new Daemon($command);
$daemon->run();

这个简单 Daemon 类虽然能实现对单个脚本进行重启守护,但是,如果我们有许多个脚本同时需要守护的,这个 Daemon 类显然是不能够满足需求的。

下一篇文章 使用 swoole 实现进程的守护(二)将尝试扩展这个 Daemon 类。

相关推荐