我想检查一下 Gearman 守护进程是否正在运行。然后才运行任务,这样我的应用程序就不会崩溃。

这是我的代码:

$daemonRunning = true;

while( true )
{
    try
    {
        Yii::app()->gearman->client->ping( true );

        if ( $daemonRunning === false )
        {
            echo "Daemon back online. Starting signature process...\n";
        }

        Yii::app()->gearman->client->runTasks();
    }
    catch( GearmanException $e )
    {
        echo "Daemon appears to be down. Waiting for it to come back up...\n";
        $daemonRunning = false;
    }
    sleep(1);
}

但问题是 ping 并没有抛出异常,而是抛出了一个致命错误:
PHP Error[2]: GearmanClient::ping(): flush(GEARMAN_COULD_NOT_CONNECT) 127.0.0.1:4730 -> libgearman/connection.cc:673

尽管奇怪的是,如果我删除 ping 并仅使用 runTasks ,则会引发异常。

相关:

当进程正在运行时 Gearman 守护程序关闭时如何处理错误?当我关闭 Gearman 守护程序时,我从 PHP 收到以下错误:
php: libgearman/universal.cc:481: gearman_return_t connection_loop(gearman_universal_st&, const gearman_packet_st&, Check&): Assertion `&con->_packet == universal.packet_list' failed.
Aborted (core dumped)

最佳答案

在最基本的情况下,可以通过命令行使用以下命令检查 Gearman 服务器的状态:
(echo status ; sleep 0.1) | nc 127.0.0.1 4730 -w 1
但是,作为 noted in this question ,您可以使用 fsocketopen 来获取齿轮服务器的状态。

 // Taken from https://stackoverflow.com/questions/2752431/any-way-to-access-gearman-administration
class Waps_Gearman_Server {

    /**
     * @var string
     */
    protected $host = "127.0.0.1";
    /**
     * @var int
     */
    protected $port = 4730;

    /**
     * @param string $host
     * @param int $port
     */
    public function __construct($host=null,$port=null){
        if( !is_null($host) ){
            $this->host = $host;
        }
        if( !is_null($port) ){
            $this->port = $port;
        }
    }

    /**
     * @return array | null
     */
    public function getStatus(){
        $status = null;
        $handle = fsockopen($this->host,$this->port,$errorNumber,$errorString,30);
        if($handle!=null){
            fwrite($handle,"status\n");
            while (!feof($handle)) {
                $line = fgets($handle, 4096);
                if( $line==".\n"){
                    break;
                }
                if( preg_match("~^(.*)[ \t](\d+)[ \t](\d+)[ \t](\d+)~",$line,$matches) ){
                    $function = $matches[1];
                    $status['operations'][$function] = array(
                        'function' => $function,
                        'total' => $matches[2],
                        'running' => $matches[3],
                        'connectedWorkers' => $matches[4],
                    );
                }
            }
            fwrite($handle,"workers\n");
            while (!feof($handle)) {
                $line = fgets($handle, 4096);
                if( $line==".\n"){
                    break;
                }
                // FD IP-ADDRESS CLIENT-ID : FUNCTION
                if( preg_match("~^(\d+)[ \t](.*?)[ \t](.*?) : ?(.*)~",$line,$matches) ){
                    $fd = $matches[1];
                    $status['connections'][$fd] = array(
                        'fd' => $fd,
                        'ip' => $matches[2],
                        'id' => $matches[3],
                        'function' => $matches[4],
                    );
                }
            }
            fclose($handle);
        }

        return $status;
    }

}

关于你的第二个问题,当连接在工作中丢失时,我从来没有能够恢复一个齿轮 worker 。您基本上必须终止运行客户端工作程序的整个进程,然后让主管接管并重新启动另一个工作进程。
Gearman 的客户工作人员应该是非常短暂的。您应该定期监视它们的内存使用情况并自动杀死它们。因此,如果您遇到段错误/核心转储,杀死工作人员是完全正常的。

如前所述,您可以使用 Supervisor 自动启动备份您的工作人员。也检查一下,它解释了如何获得 Gearman Clients working with Supervisor

关于php - 测试 Gearman 守护进程是否正在运行,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/13229566/

10-10 14:10