126 lines
3.6 KiB
PHP
126 lines
3.6 KiB
PHP
<?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\Mailer\Transport;
|
|
|
|
use Symfony\Component\Mailer\Envelope;
|
|
use Symfony\Component\Mailer\Exception\TransportException;
|
|
use Symfony\Component\Mailer\Exception\TransportExceptionInterface;
|
|
use Symfony\Component\Mailer\SentMessage;
|
|
use Symfony\Component\Mime\RawMessage;
|
|
|
|
/**
|
|
* Uses several Transports using a round robin algorithm.
|
|
*
|
|
* @author Fabien Potencier <fabien@symfony.com>
|
|
*/
|
|
class RoundRobinTransport implements TransportInterface
|
|
{
|
|
/**
|
|
* @var \SplObjectStorage<TransportInterface, float>
|
|
*/
|
|
private \SplObjectStorage $deadTransports;
|
|
private array $transports = [];
|
|
private int $retryPeriod;
|
|
private int $cursor = -1;
|
|
|
|
/**
|
|
* @param TransportInterface[] $transports
|
|
*/
|
|
public function __construct(array $transports, int $retryPeriod = 60)
|
|
{
|
|
if (!$transports) {
|
|
throw new TransportException(sprintf('"%s" must have at least one transport configured.', static::class));
|
|
}
|
|
|
|
$this->transports = $transports;
|
|
$this->deadTransports = new \SplObjectStorage();
|
|
$this->retryPeriod = $retryPeriod;
|
|
}
|
|
|
|
public function send(RawMessage $message, ?Envelope $envelope = null): ?SentMessage
|
|
{
|
|
$exception = null;
|
|
|
|
while ($transport = $this->getNextTransport()) {
|
|
try {
|
|
return $transport->send($message, $envelope);
|
|
} catch (TransportExceptionInterface $e) {
|
|
$exception ??= new TransportException('All transports failed.');
|
|
$exception->appendDebug(sprintf("Transport \"%s\": %s\n", $transport, $e->getDebug()));
|
|
$this->deadTransports[$transport] = microtime(true);
|
|
}
|
|
}
|
|
|
|
throw $exception ?? new TransportException('No transports found.');
|
|
}
|
|
|
|
public function __toString(): string
|
|
{
|
|
return $this->getNameSymbol().'('.implode(' ', array_map('strval', $this->transports)).')';
|
|
}
|
|
|
|
/**
|
|
* Rotates the transport list around and returns the first instance.
|
|
*/
|
|
protected function getNextTransport(): ?TransportInterface
|
|
{
|
|
if (-1 === $this->cursor) {
|
|
$this->cursor = $this->getInitialCursor();
|
|
}
|
|
|
|
$cursor = $this->cursor;
|
|
while (true) {
|
|
$transport = $this->transports[$cursor];
|
|
|
|
if (!$this->isTransportDead($transport)) {
|
|
break;
|
|
}
|
|
|
|
if ((microtime(true) - $this->deadTransports[$transport]) > $this->retryPeriod) {
|
|
$this->deadTransports->detach($transport);
|
|
|
|
break;
|
|
}
|
|
|
|
if ($this->cursor === $cursor = $this->moveCursor($cursor)) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
$this->cursor = $this->moveCursor($cursor);
|
|
|
|
return $transport;
|
|
}
|
|
|
|
protected function isTransportDead(TransportInterface $transport): bool
|
|
{
|
|
return $this->deadTransports->contains($transport);
|
|
}
|
|
|
|
protected function getInitialCursor(): int
|
|
{
|
|
// the cursor initial value is randomized so that
|
|
// when are not in a daemon, we are still rotating the transports
|
|
return mt_rand(0, \count($this->transports) - 1);
|
|
}
|
|
|
|
protected function getNameSymbol(): string
|
|
{
|
|
return 'roundrobin';
|
|
}
|
|
|
|
private function moveCursor(int $cursor): int
|
|
{
|
|
return ++$cursor >= \count($this->transports) ? 0 : $cursor;
|
|
}
|
|
}
|