2019-03-28 15:34:48 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file is part of the Symfony package.
|
|
|
|
*
|
|
|
|
* (c) Fabien Potencier <fabien@symfony.com>
|
|
|
|
*
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Symfony\Component\Messenger\Worker;
|
|
|
|
|
|
|
|
use Psr\Cache\CacheItemPoolInterface;
|
|
|
|
use Psr\Log\LoggerInterface;
|
|
|
|
use Symfony\Component\Messenger\Envelope;
|
|
|
|
use Symfony\Component\Messenger\WorkerInterface;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @author Ryan Weaver <ryan@symfonycasts.com>
|
|
|
|
*
|
|
|
|
* @experimental in 4.3
|
|
|
|
*/
|
|
|
|
class StopWhenRestartSignalIsReceived implements WorkerInterface
|
|
|
|
{
|
|
|
|
public const RESTART_REQUESTED_TIMESTAMP_KEY = 'workers.restart_requested_timestamp';
|
|
|
|
|
|
|
|
private $decoratedWorker;
|
|
|
|
private $cachePool;
|
|
|
|
private $logger;
|
|
|
|
|
|
|
|
public function __construct(WorkerInterface $decoratedWorker, CacheItemPoolInterface $cachePool, LoggerInterface $logger = null)
|
|
|
|
{
|
|
|
|
$this->decoratedWorker = $decoratedWorker;
|
|
|
|
$this->cachePool = $cachePool;
|
|
|
|
$this->logger = $logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function run(array $options = [], callable $onHandledCallback = null): void
|
|
|
|
{
|
2019-03-31 18:00:04 +01:00
|
|
|
$workerStartedAt = microtime(true);
|
2019-03-28 15:34:48 +00:00
|
|
|
|
2019-03-31 18:00:04 +01:00
|
|
|
$this->decoratedWorker->run($options, function (?Envelope $envelope) use ($onHandledCallback, $workerStartedAt) {
|
2019-03-28 15:34:48 +00:00
|
|
|
if (null !== $onHandledCallback) {
|
|
|
|
$onHandledCallback($envelope);
|
|
|
|
}
|
|
|
|
|
2019-03-31 18:00:04 +01:00
|
|
|
if ($this->shouldRestart($workerStartedAt)) {
|
2019-03-28 15:34:48 +00:00
|
|
|
$this->stop();
|
|
|
|
if (null !== $this->logger) {
|
|
|
|
$this->logger->info('Worker stopped because a restart was requested.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public function stop(): void
|
|
|
|
{
|
|
|
|
$this->decoratedWorker->stop();
|
|
|
|
}
|
|
|
|
|
2019-05-06 03:31:50 +01:00
|
|
|
private function shouldRestart(float $workerStartedAt): bool
|
2019-03-28 15:34:48 +00:00
|
|
|
{
|
|
|
|
$cacheItem = $this->cachePool->getItem(self::RESTART_REQUESTED_TIMESTAMP_KEY);
|
2019-03-31 18:00:04 +01:00
|
|
|
|
2019-03-28 15:34:48 +00:00
|
|
|
if (!$cacheItem->isHit()) {
|
|
|
|
// no restart has ever been scheduled
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $workerStartedAt < $cacheItem->get();
|
|
|
|
}
|
|
|
|
}
|